3 DHCP options parsing and reassembly. */
6 * Copyright (c) 2004 by Internet Systems Consortium, Inc. ("ISC")
7 * Copyright (c) 1995-2003 by Internet Software Consortium
9 * Permission to use, copy, modify, and distribute this software for any
10 * purpose with or without fee is hereby granted, provided that the above
11 * copyright notice and this permission notice appear in all copies.
13 * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES
14 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
15 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR
16 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
17 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
18 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT
19 * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
21 * Internet Systems Consortium, Inc.
23 * Redwood City, CA 94063
27 * This software has been written for Internet Systems Consortium
28 * by Ted Lemon in cooperation with Vixie Enterprises and Nominum, Inc.
29 * To learn more about Internet Systems Consortium, see
30 * ``http://www.isc.org/''. To learn more about Vixie Enterprises,
31 * see ``http://www.vix.com''. To learn more about Nominum, Inc., see
32 * ``http://www.nominum.com''.
36 static char copyright
[] =
37 "$Id$ Copyright (c) 2004 Internet Systems Consortium. All rights reserved.\n";
40 #define DHCP_OPTION_DATA
42 #include <omapip/omapip_p.h>
44 struct option
*vendor_cfg_option
;
46 static void do_option_set
PROTO ((pair
*,
47 struct option_cache
*,
50 /* Parse all available options out of the specified packet. */
52 int parse_options (packet
)
53 struct packet
*packet
;
56 struct option_cache
*op
= (struct option_cache
*)0;
58 /* Allocate a new option state. */
59 if (!option_state_allocate (&packet
-> options
, MDL
)) {
60 packet
-> options_valid
= 0;
64 /* If we don't see the magic cookie, there's nothing to parse. */
65 if (memcmp (packet
-> raw
-> options
, DHCP_OPTIONS_COOKIE
, 4)) {
66 packet
-> options_valid
= 0;
70 /* Go through the options field, up to the end of the packet
72 if (!parse_option_buffer (packet
-> options
,
73 &packet
-> raw
-> options
[4],
74 (packet
-> packet_length
-
75 DHCP_FIXED_NON_UDP
- 4),
79 /* If we parsed a DHCP Option Overload option, parse more
80 options out of the buffer(s) containing them. */
81 if (packet
-> options_valid
&&
82 (op
= lookup_option (&dhcp_universe
, packet
-> options
,
83 DHO_DHCP_OPTION_OVERLOAD
))) {
84 if (op
-> data
.data
[0] & 1) {
85 if (!parse_option_buffer
87 (unsigned char *)packet
-> raw
-> file
,
88 sizeof packet
-> raw
-> file
,
92 if (op
-> data
.data
[0] & 2) {
93 if (!parse_option_buffer
95 (unsigned char *)packet
-> raw
-> sname
,
96 sizeof packet
-> raw
-> sname
,
101 packet
-> options_valid
= 1;
105 /* Parse options out of the specified buffer, storing addresses of option
106 values in packet -> options and setting packet -> options_valid if no
107 errors are encountered. */
109 int parse_option_buffer (options
, buffer
, length
, universe
)
110 struct option_state
*options
;
111 const unsigned char *buffer
;
113 struct universe
*universe
;
116 const unsigned char *end
= buffer
+ length
;
117 unsigned len
, offset
;
119 struct option_cache
*op
= (struct option_cache
*)0;
120 struct buffer
*bp
= (struct buffer
*)0;
122 if (!buffer_allocate (&bp
, length
, MDL
)) {
123 log_error ("no memory for option buffer.");
126 memcpy (bp
-> data
, buffer
, length
);
128 for (offset
= 0; buffer
[offset
] != DHO_END
&& offset
< length
; ) {
129 code
= buffer
[offset
];
130 /* Pad options don't have a length - just skip them. */
131 if (code
== DHO_PAD
) {
136 /* Don't look for length if the buffer isn't that big. */
137 if (offset
+ 2 > length
) {
142 /* All other fields (except end, see above) have a
144 len
= buffer
[offset
+ 1];
146 /* If the length is outrageous, the options are bad. */
147 if (offset
+ len
+ 2 > length
) {
149 log_error ("parse_option_buffer: option %s (%d) %s.",
150 dhcp_options
[code
].name
, len
,
151 "larger than buffer");
152 buffer_dereference (&bp
, MDL
);
156 /* If the option contains an encapsulation, parse it. If
157 the parse fails, or the option isn't an encapsulation (by
158 far the most common case), or the option isn't entirely
159 an encapsulation, keep the raw data as well. */
160 if (universe
-> options
[code
] &&
161 !((universe
-> options
[code
] -> format
[0] == 'e' ||
162 universe
-> options
[code
] -> format
[0] == 'E') &&
163 (parse_encapsulated_suboptions
164 (options
, universe
-> options
[code
],
165 buffer
+ offset
+ 2, len
,
166 universe
, (const char *)0)))) {
167 op
= lookup_option (universe
, options
, code
);
169 struct data_string
new;
170 memset (&new, 0, sizeof new);
171 if (!buffer_allocate (&new.buffer
, op
-> data
.len
+ len
,
173 log_error ("parse_option_buffer: No memory.");
176 memcpy (new.buffer
-> data
, op
-> data
.data
,
178 memcpy (&new.buffer
-> data
[op
-> data
.len
],
179 &bp
-> data
[offset
+ 2], len
);
180 new.len
= op
-> data
.len
+ len
;
181 new.data
= new.buffer
-> data
;
182 data_string_forget (&op
-> data
, MDL
);
183 data_string_copy (&op
-> data
, &new, MDL
);
184 data_string_forget (&new, MDL
);
186 save_option_buffer (universe
, options
, bp
,
187 &bp
-> data
[offset
+ 2], len
,
188 universe
-> options
[code
], 1);
193 buffer_dereference (&bp
, MDL
);
197 /* If an option in an option buffer turns out to be an encapsulation,
198 figure out what to do. If we don't know how to de-encapsulate it,
199 or it's not well-formed, return zero; otherwise, return 1, indicating
200 that we succeeded in de-encapsulating it. */
202 struct universe
*find_option_universe (struct option
*eopt
, const char *uname
)
206 struct universe
*universe
= (struct universe
*)0;
208 /* Look for the E option in the option format. */
209 s
= strchr (eopt
-> format
, 'E');
211 log_error ("internal encapsulation format error 1.");
214 /* Look for the universe name in the option format. */
215 t
= strchr (++s
, '.');
216 /* If there was no trailing '.', or there's something after the
217 trailing '.', the option is bogus and we can't use it. */
219 log_error ("internal encapsulation format error 2.");
222 if (t
== s
&& uname
) {
223 for (i
= 0; i
< universe_count
; i
++) {
224 if (!strcmp (universes
[i
] -> name
, uname
)) {
225 universe
= universes
[i
];
230 for (i
= 0; i
< universe_count
; i
++) {
231 if (strlen (universes
[i
] -> name
) == t
- s
&&
232 !memcmp (universes
[i
] -> name
,
233 s
, (unsigned)(t
- s
))) {
234 universe
= universes
[i
];
242 /* If an option in an option buffer turns out to be an encapsulation,
243 figure out what to do. If we don't know how to de-encapsulate it,
244 or it's not well-formed, return zero; otherwise, return 1, indicating
245 that we succeeded in de-encapsulating it. */
247 int parse_encapsulated_suboptions (struct option_state
*options
,
249 const unsigned char *buffer
,
250 unsigned len
, struct universe
*eu
,
254 struct universe
*universe
= find_option_universe (eopt
, uname
);
256 /* If we didn't find the universe, we can't do anything with it
257 right now (e.g., we can't decode vendor options until we've
258 decoded the packet and executed the scopes that it matches). */
262 /* If we don't have a decoding function for it, we can't decode
264 if (!universe
-> decode
)
267 i
= (*universe
-> decode
) (options
, buffer
, len
, universe
);
269 /* If there is stuff before the suboptions, we have to keep it. */
270 if (eopt
-> format
[0] != 'E')
272 /* Otherwise, return the status of the decode function. */
276 int fqdn_universe_decode (struct option_state
*options
,
277 const unsigned char *buffer
,
278 unsigned length
, struct universe
*u
)
281 struct buffer
*bp
= (struct buffer
*)0;
283 /* FQDN options have to be at least four bytes long. */
287 /* Save the contents of the option in a buffer. */
288 if (!buffer_allocate (&bp
, length
+ 4, MDL
)) {
289 log_error ("no memory for option buffer.");
292 memcpy (&bp
-> data
[3], buffer
+ 1, length
- 1);
294 if (buffer
[0] & 4) /* encoded */
298 if (!save_option_buffer (&fqdn_universe
, options
, bp
,
300 &fqdn_options
[FQDN_ENCODED
], 0)) {
302 buffer_dereference (&bp
, MDL
);
306 if (buffer
[0] & 1) /* server-update */
310 if (buffer
[0] & 2) /* no-client-update */
315 /* XXX Ideally we should store the name in DNS format, so if the
316 XXX label isn't in DNS format, we convert it to DNS format,
317 XXX rather than converting labels specified in DNS format to
318 XXX the plain ASCII representation. But that's hard, so
321 /* Not encoded using DNS format? */
322 if (!bp
-> data
[0]) {
325 /* Some broken clients NUL-terminate this option. */
326 if (buffer
[length
- 1] == 0) {
331 /* Determine the length of the hostname component of the
332 name. If the name contains no '.' character, it
333 represents a non-qualified label. */
334 for (i
= 3; i
< length
&& buffer
[i
] != '.'; i
++);
337 /* Note: If the client sends a FQDN, the first '.' will
338 be used as a NUL terminator for the hostname. */
340 if (!save_option_buffer (&fqdn_universe
, options
, bp
,
342 &fqdn_options
[FQDN_HOSTNAME
],
345 /* Note: If the client sends a single label, the
346 FQDN_DOMAINNAME option won't be set. */
347 if (length
> 4 + i
&&
348 !save_option_buffer (&fqdn_universe
, options
, bp
,
349 &bp
-> data
[6 + i
], length
- 4 - i
,
350 &fqdn_options
[FQDN_DOMAINNAME
], 1))
352 /* Also save the whole name. */
354 if (!save_option_buffer (&fqdn_universe
, options
, bp
,
355 &bp
-> data
[5], length
- 3,
356 &fqdn_options
[FQDN_FQDN
], 1))
360 unsigned total_len
= 0;
361 unsigned first_len
= 0;
367 while (s
< &bp
-> data
[0] + length
+ 2) {
370 log_info ("fancy bits in fqdn option");
377 if (s
+ len
> &bp
-> data
[0] + length
+ 3) {
378 log_info ("fqdn tag longer than buffer");
382 if (first_len
== 0) {
388 total_len
+= len
+ 1;
391 /* We wind up with a length that's one too many because
392 we shouldn't increment for the last label, but there's
393 no way to tell we're at the last label until we exit
399 first_len
= total_len
;
403 !save_option_buffer (&fqdn_universe
, options
, bp
,
404 &bp
-> data
[6], first_len
,
405 &fqdn_options
[FQDN_HOSTNAME
], 0))
407 if (total_len
> 0 && first_len
!= total_len
) {
408 if (!save_option_buffer
409 (&fqdn_universe
, options
, bp
,
410 &bp
-> data
[6 + first_len
], total_len
- first_len
,
411 &fqdn_options
[FQDN_DOMAINNAME
], 1))
415 if (!save_option_buffer (&fqdn_universe
, options
, bp
,
416 &bp
-> data
[6], total_len
,
417 &fqdn_options
[FQDN_FQDN
], 1))
421 if (!save_option_buffer (&fqdn_universe
, options
, bp
,
423 &fqdn_options
[FQDN_NO_CLIENT_UPDATE
], 0))
425 if (!save_option_buffer (&fqdn_universe
, options
, bp
,
427 &fqdn_options
[FQDN_SERVER_UPDATE
], 0))
430 if (!save_option_buffer (&fqdn_universe
, options
, bp
,
432 &fqdn_options
[FQDN_RCODE1
], 0))
434 if (!save_option_buffer (&fqdn_universe
, options
, bp
,
436 &fqdn_options
[FQDN_RCODE2
], 0))
439 buffer_dereference (&bp
, MDL
);
443 /* cons options into a big buffer, and then split them out into the
444 three seperate buffers if needed. This allows us to cons up a set
445 of vendor options using the same routine. */
447 int cons_options (inpacket
, outpacket
, lease
, client_state
,
448 mms
, in_options
, cfg_options
,
449 scope
, overload
, terminate
, bootpp
, prl
, vuname
)
450 struct packet
*inpacket
;
451 struct dhcp_packet
*outpacket
;
453 struct client_state
*client_state
;
455 struct option_state
*in_options
;
456 struct option_state
*cfg_options
;
457 struct binding_scope
**scope
;
458 int overload
; /* Overload flags that may be set. */
461 struct data_string
*prl
;
464 #define PRIORITY_COUNT 300
465 unsigned priority_list
[PRIORITY_COUNT
];
467 unsigned char buffer
[4096]; /* Really big buffer... */
468 unsigned main_buffer_size
;
469 unsigned mainbufix
, bufix
, agentix
;
472 unsigned option_size
;
475 struct option_cache
*op
;
476 struct data_string ds
;
481 int ofbuf1
=0, ofbuf2
=0;
483 memset (&ds
, 0, sizeof ds
);
485 /* If there's a Maximum Message Size option in the incoming packet
486 and no alternate maximum message size has been specified, take the
487 one in the packet. */
490 (op
= lookup_option (&dhcp_universe
, inpacket
-> options
,
491 DHO_DHCP_MAX_MESSAGE_SIZE
))) {
492 evaluate_option_cache (&ds
, inpacket
,
493 lease
, client_state
, in_options
,
494 cfg_options
, scope
, op
, MDL
);
495 if (ds
.len
>= sizeof (u_int16_t
)) {
496 i
= getUShort (ds
.data
);
498 if(!mms
|| (i
< mms
))
501 data_string_forget (&ds
, MDL
);
504 /* If the client has provided a maximum DHCP message size,
505 use that; otherwise, if it's BOOTP, only 64 bytes; otherwise
506 use up to the minimum IP MTU size (576 bytes). */
507 /* XXX if a BOOTP client specifies a max message size, we will
511 main_buffer_size
= mms
- DHCP_FIXED_LEN
;
513 /* Enforce a minimum packet size... */
514 if (main_buffer_size
< (576 - DHCP_FIXED_LEN
))
515 main_buffer_size
= 576 - DHCP_FIXED_LEN
;
519 inpacket
-> packet_length
- DHCP_FIXED_LEN
;
520 if (main_buffer_size
< 64)
521 main_buffer_size
= 64;
523 main_buffer_size
= 64;
525 main_buffer_size
= 576 - DHCP_FIXED_LEN
;
527 /* Set a hard limit at the size of the output buffer. */
528 if (main_buffer_size
> sizeof buffer
)
529 main_buffer_size
= sizeof buffer
;
531 /* Preload the option priority list with mandatory options. */
533 priority_list
[priority_len
++] = DHO_DHCP_MESSAGE_TYPE
;
534 priority_list
[priority_len
++] = DHO_DHCP_SERVER_IDENTIFIER
;
535 priority_list
[priority_len
++] = DHO_DHCP_LEASE_TIME
;
536 priority_list
[priority_len
++] = DHO_DHCP_MESSAGE
;
537 priority_list
[priority_len
++] = DHO_DHCP_REQUESTED_ADDRESS
;
538 priority_list
[priority_len
++] = DHO_FQDN
;
540 if (prl
&& prl
-> len
> 0) {
541 if ((op
= lookup_option (&dhcp_universe
, cfg_options
,
542 DHO_SUBNET_SELECTION
))) {
543 if (priority_len
< PRIORITY_COUNT
)
544 priority_list
[priority_len
++] =
545 DHO_SUBNET_SELECTION
;
548 data_string_truncate (prl
, (PRIORITY_COUNT
- priority_len
));
550 for (i
= 0; i
< prl
-> len
; i
++) {
551 /* Prevent client from changing order of delivery
552 of relay agent information option. */
553 if (prl
-> data
[i
] != DHO_DHCP_AGENT_OPTIONS
)
554 priority_list
[priority_len
++] =
558 /* First, hardcode some more options that ought to be
560 priority_list
[priority_len
++] = DHO_SUBNET_MASK
;
561 priority_list
[priority_len
++] = DHO_ROUTERS
;
562 priority_list
[priority_len
++] = DHO_DOMAIN_NAME_SERVERS
;
563 priority_list
[priority_len
++] = DHO_HOST_NAME
;
565 /* Append a list of the standard DHCP options from the
566 standard DHCP option space. Actually, if a site
567 option space hasn't been specified, we wind up
568 treating the dhcp option space as the site option
569 space, and the first for loop is skipped, because
570 it's slightly more general to do it this way,
571 taking the 1Q99 DHCP futures work into account. */
572 if (cfg_options
-> site_code_min
) {
573 for (i
= 0; i
< OPTION_HASH_SIZE
; i
++) {
574 hash
= cfg_options
-> universes
[dhcp_universe
.index
];
576 for (pp
= hash
[i
]; pp
; pp
= pp
-> cdr
) {
577 op
= (struct option_cache
*)(pp
-> car
);
578 if (op
-> option
-> code
<
579 cfg_options
-> site_code_min
&&
580 priority_len
< PRIORITY_COUNT
&&
581 (op
-> option
-> code
!=
582 DHO_DHCP_AGENT_OPTIONS
))
583 priority_list
[priority_len
++] =
584 op
-> option
-> code
;
590 /* Now cycle through the site option space, or if there
591 is no site option space, we'll be cycling through the
592 dhcp option space. */
593 for (i
= 0; i
< OPTION_HASH_SIZE
; i
++) {
594 hash
= (cfg_options
-> universes
595 [cfg_options
-> site_universe
]);
597 for (pp
= hash
[i
]; pp
; pp
= pp
-> cdr
) {
598 op
= (struct option_cache
*)(pp
-> car
);
599 if (op
-> option
-> code
>=
600 cfg_options
-> site_code_min
&&
601 priority_len
< PRIORITY_COUNT
&&
602 (op
-> option
-> code
!=
603 DHO_DHCP_AGENT_OPTIONS
))
604 priority_list
[priority_len
++] =
605 op
-> option
-> code
;
609 /* Now go through all the universes for which options
610 were set and see if there are encapsulations for
611 them; if there are, put the encapsulation options
612 on the priority list as well. */
613 for (i
= 0; i
< cfg_options
-> universe_count
; i
++) {
614 if (cfg_options
-> universes
[i
] &&
615 universes
[i
] -> enc_opt
&&
616 priority_len
< PRIORITY_COUNT
&&
617 universes
[i
] -> enc_opt
-> universe
== &dhcp_universe
)
619 if (universes
[i
] -> enc_opt
-> code
!=
620 DHO_DHCP_AGENT_OPTIONS
)
621 priority_list
[priority_len
++] =
622 universes
[i
] -> enc_opt
-> code
;
626 /* The vendor option space can't stand on its own, so always
627 add it to the list. */
628 if (priority_len
< PRIORITY_COUNT
)
629 priority_list
[priority_len
++] =
630 DHO_VENDOR_ENCAPSULATED_OPTIONS
;
633 /* Figure out the overload buffer offset(s). */
635 ofbuf1
= main_buffer_size
- 4;
637 ofbuf2
= main_buffer_size
- 4 + DHCP_FILE_LEN
;
640 /* Copy the options into the big buffer... */
641 option_size
= store_options (&ocount
, buffer
,
642 (main_buffer_size
- 4 +
643 ((overload
& 1) ? DHCP_FILE_LEN
: 0) +
644 ((overload
& 2) ? DHCP_SNAME_LEN
: 0)),
645 inpacket
, lease
, client_state
,
646 in_options
, cfg_options
, scope
,
647 priority_list
, priority_len
,
648 ofbuf1
, ofbuf2
, terminate
, vuname
);
649 /* If store_options failed. */
650 if (option_size
== 0)
653 if (ocount
== 1 && (overload
& 1))
655 else if (ocount
== 1 && (overload
& 2))
657 else if (ocount
== 3)
663 /* Put the cookie up front... */
664 memcpy (outpacket
-> options
, DHCP_OPTIONS_COOKIE
, 4);
667 /* If we're going to have to overload, store the overload
668 option at the beginning. If we can, though, just store the
669 whole thing in the packet's option buffer and leave it at
671 memcpy (&outpacket
-> options
[mainbufix
],
672 buffer
, option_size
);
673 mainbufix
+= option_size
;
675 outpacket
-> options
[mainbufix
++] = DHO_DHCP_OPTION_OVERLOAD
;
676 outpacket
-> options
[mainbufix
++] = 1;
677 outpacket
-> options
[mainbufix
++] = overload
;
680 memcpy (outpacket
-> file
,
681 &buffer
[ofbuf1
], DHCP_FILE_LEN
);
685 memcpy (outpacket
-> sname
, &buffer
[ofbuf2
],
688 memcpy (outpacket
-> sname
, &buffer
[ofbuf1
],
694 if (mainbufix
< main_buffer_size
)
696 length
= DHCP_FIXED_NON_UDP
+ mainbufix
;
698 /* Now hack in the agent options if there are any. */
699 priority_list
[0] = DHO_DHCP_AGENT_OPTIONS
;
702 store_options (0, &outpacket
-> options
[agentix
],
703 1500 - DHCP_FIXED_LEN
- agentix
,
704 inpacket
, lease
, client_state
,
705 in_options
, cfg_options
, scope
,
706 priority_list
, priority_len
,
709 /* Tack a DHO_END option onto the packet if we need to. */
710 if (agentix
< 1500 - DHCP_FIXED_LEN
&& need_endopt
)
711 outpacket
-> options
[agentix
++] = DHO_END
;
713 /* Figure out the length. */
714 length
= DHCP_FIXED_NON_UDP
+ agentix
;
718 /* Store all the requested options into the requested buffer. */
720 int store_options (ocount
, buffer
, buflen
, packet
, lease
, client_state
,
721 in_options
, cfg_options
, scope
, priority_list
, priority_len
,
722 first_cutoff
, second_cutoff
, terminate
, vuname
)
724 unsigned char *buffer
;
726 struct packet
*packet
;
728 struct client_state
*client_state
;
729 struct option_state
*in_options
;
730 struct option_state
*cfg_options
;
731 struct binding_scope
**scope
;
732 unsigned *priority_list
;
734 unsigned first_cutoff
, second_cutoff
;
738 int bufix
= 0, six
= 0, tix
= 0;
743 struct data_string od
;
744 struct option_cache
*oc
;
748 if (first_cutoff
>= buflen
)
749 log_fatal("%s:%d:store_options: Invalid first cutoff.", MDL
);
751 bufend
= first_cutoff
;
756 if (second_cutoff
>= buflen
)
757 log_fatal("%s:%d:store_options: Invalid second cutoff.", MDL
);
759 sbufend
= second_cutoff
;
763 memset (&od
, 0, sizeof od
);
765 /* Eliminate duplicate options in the parameter request list.
766 There's got to be some clever knuthian way to do this:
767 Eliminate all but the first occurance of a value in an array
768 of values without otherwise disturbing the order of the array. */
769 for (i
= 0; i
< priority_len
- 1; i
++) {
771 for (ix
= i
+ 1; ix
< priority_len
+ tto
; ix
++) {
773 priority_list
[ix
- tto
] =
775 if (priority_list
[i
] == priority_list
[ix
]) {
782 /* Copy out the options in the order that they appear in the
784 for (i
= 0; i
< priority_len
; i
++) {
785 /* Number of bytes left to store (some may already
786 have been stored by a previous pass). */
788 int optstart
, soptstart
, toptstart
;
790 int have_encapsulation
= 0;
791 struct data_string encapsulation
;
794 memset (&encapsulation
, 0, sizeof encapsulation
);
796 /* Code for next option to try to store. */
797 code
= priority_list
[i
];
799 /* Look up the option in the site option space if the code
800 is above the cutoff, otherwise in the DHCP option space. */
801 if (code
>= cfg_options
-> site_code_min
)
802 u
= universes
[cfg_options
-> site_universe
];
806 oc
= lookup_option (u
, cfg_options
, code
);
808 /* It's an encapsulation, try to find the universe
809 to be encapsulated first, except that if it's a straight
810 encapsulation and the user has provided a value for the
811 encapsulation option, use the user-provided value. */
812 if (u
-> options
[code
] &&
813 ((u
-> options
[code
] -> format
[0] == 'E' && !oc
) ||
814 u
-> options
[code
] -> format
[0] == 'e')) {
817 struct option_cache
*tmp
;
818 struct data_string name
;
820 s
= strchr (u
-> options
[code
] -> format
, 'E');
822 t
= strchr (++s
, '.');
824 memset (&name
, 0, sizeof name
);
826 /* A zero-length universe name means the vendor
827 option space, if one is defined. */
829 if (vendor_cfg_option
) {
830 tmp
= lookup_option (vendor_cfg_option
-> universe
,
832 vendor_cfg_option
-> code
);
834 evaluate_option_cache (&name
, packet
, lease
,
840 name
.data
= (unsigned char *)s
;
841 name
.len
= strlen (s
);
844 name
.data
= (unsigned char *)s
;
848 /* If we found a universe, and there are options configured
849 for that universe, try to encapsulate it. */
852 (option_space_encapsulate
853 (&encapsulation
, packet
, lease
, client_state
,
854 in_options
, cfg_options
, scope
, &name
));
855 data_string_forget (&name
, MDL
);
860 /* In order to avoid memory leaks, we have to get to here
861 with any option cache that we allocated in tmp not being
862 referenced by tmp, and whatever option cache is referenced
863 by oc being an actual reference. lookup_option doesn't
864 generate a reference (this needs to be fixed), so the
865 preceding goop ensures that if we *didn't* generate a new
866 option cache, oc still winds up holding an actual reference. */
868 /* If no data is available for this option, skip it. */
869 if (!oc
&& !have_encapsulation
) {
873 /* Find the value of the option... */
875 evaluate_option_cache (&od
, packet
,
876 lease
, client_state
, in_options
,
877 cfg_options
, scope
, oc
, MDL
);
879 data_string_forget (&encapsulation
, MDL
);
880 data_string_forget (&od
, MDL
);
881 have_encapsulation
= 0;
886 /* We should now have a constant length for the option. */
888 if (have_encapsulation
) {
889 length
+= encapsulation
.len
;
891 data_string_copy (&od
, &encapsulation
, MDL
);
892 data_string_forget (&encapsulation
, MDL
);
894 struct buffer
*bp
= (struct buffer
*)0;
895 if (!buffer_allocate (&bp
, length
, MDL
)) {
896 option_cache_dereference (&oc
, MDL
);
897 data_string_forget (&od
, MDL
);
898 data_string_forget (&encapsulation
, MDL
);
901 memcpy (&bp
-> data
[0], od
.data
, od
.len
);
902 memcpy (&bp
-> data
[od
.len
], encapsulation
.data
,
904 data_string_forget (&od
, MDL
);
905 data_string_forget (&encapsulation
, MDL
);
906 od
.data
= &bp
-> data
[0];
907 buffer_reference (&od
.buffer
, bp
, MDL
);
908 buffer_dereference (&bp
, MDL
);
914 /* Do we add a NUL? */
915 if (terminate
&& dhcp_options
[code
].format
[0] == 't') {
922 /* Try to store the option. */
924 /* If the option's length is more than 255, we must store it
925 in multiple hunks. Store 255-byte hunks first. However,
926 in any case, if the option data will cross a buffer
927 boundary, split it across that boundary. */
940 unsigned incr
= length
;
945 /* Try to fit it in the options buffer. */
947 ((!six
&& !tix
&& (i
== priority_len
- 1) &&
948 (bufix
+ 2 + length
< bufend
)) ||
949 (bufix
+ 5 + length
< bufend
))) {
952 /* Try to fit it in the second buffer. */
953 } else if (!splitup
&& first_cutoff
&&
954 (first_cutoff
+ six
+ 3 + length
< sbufend
)) {
955 base
= &buffer
[first_cutoff
];
957 /* Try to fit it in the third buffer. */
958 } else if (!splitup
&& second_cutoff
&&
959 (second_cutoff
+ tix
+ 3 + length
< buflen
)) {
960 base
= &buffer
[second_cutoff
];
962 /* Split the option up into the remaining space. */
966 /* Use any remaining options space. */
967 if (bufix
+ 6 < bufend
) {
968 incr
= bufend
- bufix
- 5;
971 /* Use any remaining first_cutoff space. */
972 } else if (first_cutoff
&&
973 (first_cutoff
+ six
+ 4 < sbufend
)) {
974 incr
= sbufend
- (first_cutoff
+ six
) - 3;
975 base
= &buffer
[first_cutoff
];
977 /* Use any remaining second_cutoff space. */
978 } else if (second_cutoff
&&
979 (second_cutoff
+ tix
+ 4 < buflen
)) {
980 incr
= buflen
- (second_cutoff
+ tix
) - 3;
981 base
= &buffer
[second_cutoff
];
983 /* Give up, roll back this option. */
997 /* Everything looks good - copy it in! */
999 base
[*pix
+ 1] = (unsigned char)incr
;
1000 if (tto
&& incr
== length
) {
1002 memcpy (base
+ *pix
+ 2,
1003 od
.data
+ ix
, (unsigned)(incr
- 1));
1004 base
[*pix
+ 2 + incr
- 1] = 0;
1006 memcpy (base
+ *pix
+ 2,
1007 od
.data
+ ix
, (unsigned)incr
);
1013 data_string_forget (&od
, MDL
);
1016 /* If we can overload, and we have, then PAD and END those spaces. */
1017 if (first_cutoff
&& six
) {
1018 if ((first_cutoff
+ six
+ 1) < sbufend
)
1019 memset (&buffer
[first_cutoff
+ six
+ 1], DHO_PAD
,
1020 sbufend
- (first_cutoff
+ six
+ 1));
1021 else if (first_cutoff
+ six
>= sbufend
)
1022 log_fatal("Second buffer overflow in overloaded options.");
1024 buffer
[first_cutoff
+ six
] = DHO_END
;
1025 *ocount
|= 1; /* So that caller knows there's data there. */
1028 if (second_cutoff
&& tix
) {
1029 if (second_cutoff
+ tix
+ 1 < buflen
) {
1030 memset (&buffer
[second_cutoff
+ tix
+ 1], DHO_PAD
,
1031 buflen
- (second_cutoff
+ tix
+ 1));
1032 } else if (second_cutoff
+ tix
>= buflen
)
1033 log_fatal("Third buffer overflow in overloaded options.");
1035 buffer
[second_cutoff
+ tix
] = DHO_END
;
1036 *ocount
|= 2; /* So that caller knows there's data there. */
1039 if ((six
|| tix
) && (bufix
+ 3 > bufend
))
1040 log_fatal("Not enough space for option overload option.");
1045 /* Format the specified option so that a human can easily read it. */
1047 const char *pretty_print_option (option
, data
, len
, emit_commas
, emit_quotes
)
1048 struct option
*option
;
1049 const unsigned char *data
;
1054 static char optbuf
[32768]; /* XXX */
1061 struct enumeration
*enumbuf
[32];
1064 const unsigned char *dp
= data
;
1074 memset (enumbuf
, 0, sizeof enumbuf
);
1076 /* Figure out the size of the data. */
1077 for (l
= i
= 0; option
-> format
[i
]; i
++, l
++) {
1079 log_error ("%s: Extra codes in format string: %s",
1081 &(option
-> format
[i
]));
1085 fmtbuf
[l
] = option
-> format
[i
];
1086 switch (option
-> format
[i
]) {
1098 /* Skip the universe name. */
1099 while (option
-> format
[i
] &&
1100 option
-> format
[i
] != '.')
1103 for (k
= 0; k
< len
; k
++) {
1104 if (!isascii (data
[k
]) ||
1105 !isprint (data
[k
]))
1108 /* If we found no bogus characters, or the bogus
1109 character we found is a trailing NUL, it's
1110 okay to print this option as text. */
1111 if (k
== len
|| (k
+ 1 == len
&& data
[k
] == 0)) {
1130 while (option
-> format
[i
] &&
1131 option
-> format
[i
] != '.')
1134 find_enumeration (&option
-> format
[k
] + 1,
1163 log_error ("%s: garbage in format string: %s",
1165 &(option
-> format
[i
]));
1170 /* Check for too few bytes... */
1171 if (hunksize
- opthunk
> len
) {
1172 log_error ("%s: expecting at least %d bytes; got %d",
1177 /* Check for too many bytes... */
1178 if (numhunk
== -1 && hunksize
< len
)
1179 log_error ("%s: %d extra bytes",
1183 /* If this is an array, compute its size. */
1185 numhunk
= len
/ hunksize
;
1186 /* See if we got an exact number of hunks. */
1187 if (numhunk
> 0 && numhunk
* hunksize
< len
)
1188 log_error ("%s: %d extra bytes at end of array\n",
1190 len
- numhunk
* hunksize
);
1192 /* A one-hunk array prints the same as a single hunk. */
1196 /* Cycle through the array (or hunk) printing the data. */
1197 for (i
= 0; i
< numhunk
; i
++) {
1198 for (j
= 0; j
< numelem
; j
++) {
1199 switch (fmtbuf
[j
]) {
1203 for (; dp
< data
+ len
; dp
++) {
1204 if (!isascii (*dp
) ||
1206 /* Skip trailing NUL. */
1207 if (dp
+ 1 != data
+ len
||
1209 sprintf (op
, "\\%03o",
1213 } else if (*dp
== '"' ||
1227 /* pretty-printing an array of enums is
1228 going to get ugly. */
1233 if (!enumbuf
[j
] -> values
[i
].name
)
1235 if (enumbuf
[j
] -> values
[i
].value
==
1239 strcpy (op
, enumbuf
[j
] -> values
[i
].name
);
1243 foo
.s_addr
= htonl (getULong (dp
));
1244 strcpy (op
, inet_ntoa (foo
));
1248 sprintf (op
, "%ld", (long)getLong (dp
));
1252 tval
= getULong (dp
);
1254 sprintf (op
, "%s", "infinite");
1256 sprintf (op
, "%ld", tval
);
1260 (unsigned long)getULong (dp
));
1264 sprintf (op
, "%d", (int)getShort (dp
));
1268 sprintf (op
, "%d", (unsigned)getUShort (dp
));
1272 sprintf (op
, "%d", *(const char *)dp
++);
1276 sprintf (op
, "%d", *dp
++);
1279 sprintf (op
, "%x", *dp
++);
1282 strcpy (op
, *dp
++ ? "true" : "false");
1285 log_error ("Unexpected format code %c",
1289 if (dp
== data
+ len
)
1291 if (j
+ 1 < numelem
&& comma
!= ':')
1294 if (i
+ 1 < numhunk
) {
1297 if (dp
== data
+ len
)
1303 int get_option (result
, universe
, packet
, lease
, client_state
,
1304 in_options
, cfg_options
, options
, scope
, code
, file
, line
)
1305 struct data_string
*result
;
1306 struct universe
*universe
;
1307 struct packet
*packet
;
1308 struct lease
*lease
;
1309 struct client_state
*client_state
;
1310 struct option_state
*in_options
;
1311 struct option_state
*cfg_options
;
1312 struct option_state
*options
;
1313 struct binding_scope
**scope
;
1318 struct option_cache
*oc
;
1320 if (!universe
-> lookup_func
)
1322 oc
= ((*universe
-> lookup_func
) (universe
, options
, code
));
1325 if (!evaluate_option_cache (result
, packet
, lease
, client_state
,
1326 in_options
, cfg_options
, scope
, oc
,
1332 void set_option (universe
, options
, option
, op
)
1333 struct universe
*universe
;
1334 struct option_state
*options
;
1335 struct option_cache
*option
;
1336 enum statement_op op
;
1338 struct option_cache
*oc
, *noc
;
1343 case eval_statement
:
1344 case break_statement
:
1346 log_error ("bogus statement type in do_option_set.");
1349 case default_option_statement
:
1350 oc
= lookup_option (universe
, options
,
1351 option
-> option
-> code
);
1354 save_option (universe
, options
, option
);
1357 case supersede_option_statement
:
1358 case send_option_statement
:
1359 /* Install the option, replacing any existing version. */
1360 save_option (universe
, options
, option
);
1363 case append_option_statement
:
1364 case prepend_option_statement
:
1365 oc
= lookup_option (universe
, options
,
1366 option
-> option
-> code
);
1368 save_option (universe
, options
, option
);
1371 /* If it's not an expression, make it into one. */
1372 if (!oc
-> expression
&& oc
-> data
.len
) {
1373 if (!expression_allocate (&oc
-> expression
, MDL
)) {
1374 log_error ("Can't allocate const expression.");
1377 oc
-> expression
-> op
= expr_const_data
;
1379 (&oc
-> expression
-> data
.const_data
,
1381 data_string_forget (&oc
-> data
, MDL
);
1383 noc
= (struct option_cache
*)0;
1384 if (!option_cache_allocate (&noc
, MDL
))
1386 if (op
== append_option_statement
) {
1387 if (!make_concat (&noc
-> expression
,
1389 option
-> expression
)) {
1390 option_cache_dereference (&noc
, MDL
);
1394 if (!make_concat (&noc
-> expression
,
1395 option
-> expression
,
1396 oc
-> expression
)) {
1397 option_cache_dereference (&noc
, MDL
);
1401 noc
-> option
= oc
-> option
;
1402 save_option (universe
, options
, noc
);
1403 option_cache_dereference (&noc
, MDL
);
1408 struct option_cache
*lookup_option (universe
, options
, code
)
1409 struct universe
*universe
;
1410 struct option_state
*options
;
1414 return (struct option_cache
*)0;
1415 if (universe
-> lookup_func
)
1416 return (*universe
-> lookup_func
) (universe
, options
, code
);
1418 log_error ("can't look up options in %s space.",
1420 return (struct option_cache
*)0;
1423 struct option_cache
*lookup_hashed_option (universe
, options
, code
)
1424 struct universe
*universe
;
1425 struct option_state
*options
;
1432 /* Make sure there's a hash table. */
1433 if (universe
-> index
>= options
-> universe_count
||
1434 !(options
-> universes
[universe
-> index
]))
1435 return (struct option_cache
*)0;
1437 hash
= options
-> universes
[universe
-> index
];
1439 hashix
= compute_option_hash (code
);
1440 for (bptr
= hash
[hashix
]; bptr
; bptr
= bptr
-> cdr
) {
1441 if (((struct option_cache
*)(bptr
-> car
)) -> option
-> code
==
1443 return (struct option_cache
*)(bptr
-> car
);
1445 return (struct option_cache
*)0;
1448 int save_option_buffer (struct universe
*universe
,
1449 struct option_state
*options
,
1451 unsigned char *buffer
, unsigned length
,
1452 struct option
*option
, int tp
)
1454 struct buffer
*lbp
= (struct buffer
*)0;
1455 struct option_cache
*op
= (struct option_cache
*)0;
1457 if (!option_cache_allocate (&op
, MDL
)) {
1458 log_error ("No memory for option %s.%s.",
1464 /* If we weren't passed a buffer in which the data are saved and
1465 refcounted, allocate one now. */
1467 if (!buffer_allocate (&lbp
, length
, MDL
)) {
1468 log_error ("no memory for option buffer.");
1470 option_cache_dereference (&op
, MDL
);
1473 memcpy (lbp
-> data
, buffer
, length
+ tp
);
1475 buffer
= &bp
-> data
[0]; /* Refer to saved buffer. */
1478 /* Reference buffer copy to option cache. */
1479 op
-> data
.buffer
= (struct buffer
*)0;
1480 buffer_reference (&op
-> data
.buffer
, bp
, MDL
);
1482 /* Point option cache into buffer. */
1483 op
-> data
.data
= buffer
;
1484 op
-> data
.len
= length
;
1487 /* NUL terminate (we can get away with this because we (or
1488 the caller!) allocated one more than the buffer size, and
1489 because the byte following the end of an option is always
1490 the code of the next option, which the caller is getting
1491 out of the *original* buffer. */
1492 buffer
[length
] = 0;
1493 op
-> data
.terminated
= 1;
1495 op
-> data
.terminated
= 0;
1497 op
-> option
= option
;
1499 /* Now store the option. */
1500 save_option (universe
, options
, op
);
1502 /* And let go of our reference. */
1503 option_cache_dereference (&op
, MDL
);
1508 void save_option (struct universe
*universe
,
1509 struct option_state
*options
, struct option_cache
*oc
)
1511 if (universe
-> save_func
)
1512 (*universe
-> save_func
) (universe
, options
, oc
);
1514 log_error ("can't store options in %s space.",
1518 void save_hashed_option (universe
, options
, oc
)
1519 struct universe
*universe
;
1520 struct option_state
*options
;
1521 struct option_cache
*oc
;
1525 pair
*hash
= options
-> universes
[universe
-> index
];
1527 if (oc
-> refcnt
== 0)
1530 /* Compute the hash. */
1531 hashix
= compute_option_hash (oc
-> option
-> code
);
1533 /* If there's no hash table, make one. */
1535 hash
= (pair
*)dmalloc (OPTION_HASH_SIZE
* sizeof *hash
, MDL
);
1537 log_error ("no memory to store %s.%s",
1538 universe
-> name
, oc
-> option
-> name
);
1541 memset (hash
, 0, OPTION_HASH_SIZE
* sizeof *hash
);
1542 options
-> universes
[universe
-> index
] = (VOIDPTR
)hash
;
1544 /* Try to find an existing option matching the new one. */
1545 for (bptr
= hash
[hashix
]; bptr
; bptr
= bptr
-> cdr
) {
1546 if (((struct option_cache
*)
1547 (bptr
-> car
)) -> option
-> code
==
1548 oc
-> option
-> code
)
1552 /* If we find one, dereference it and put the new one
1555 option_cache_dereference
1556 ((struct option_cache
**)&bptr
-> car
, MDL
);
1557 option_cache_reference
1558 ((struct option_cache
**)&bptr
-> car
,
1564 /* Otherwise, just put the new one at the head of the list. */
1565 bptr
= new_pair (MDL
);
1567 log_error ("No memory for option_cache reference.");
1570 bptr
-> cdr
= hash
[hashix
];
1572 option_cache_reference ((struct option_cache
**)&bptr
-> car
, oc
, MDL
);
1573 hash
[hashix
] = bptr
;
1576 void delete_option (universe
, options
, code
)
1577 struct universe
*universe
;
1578 struct option_state
*options
;
1581 if (universe
-> delete_func
)
1582 (*universe
-> delete_func
) (universe
, options
, code
);
1584 log_error ("can't delete options from %s space.",
1588 void delete_hashed_option (universe
, options
, code
)
1589 struct universe
*universe
;
1590 struct option_state
*options
;
1594 pair bptr
, prev
= (pair
)0;
1595 pair
*hash
= options
-> universes
[universe
-> index
];
1597 /* There may not be any options in this space. */
1601 /* Try to find an existing option matching the new one. */
1602 hashix
= compute_option_hash (code
);
1603 for (bptr
= hash
[hashix
]; bptr
; bptr
= bptr
-> cdr
) {
1604 if (((struct option_cache
*)(bptr
-> car
)) -> option
-> code
1609 /* If we found one, wipe it out... */
1612 prev
-> cdr
= bptr
-> cdr
;
1614 hash
[hashix
] = bptr
-> cdr
;
1615 option_cache_dereference
1616 ((struct option_cache
**)(&bptr
-> car
), MDL
);
1617 free_pair (bptr
, MDL
);
1621 extern struct option_cache
*free_option_caches
; /* XXX */
1623 int option_cache_dereference (ptr
, file
, line
)
1624 struct option_cache
**ptr
;
1628 if (!ptr
|| !*ptr
) {
1629 log_error ("Null pointer in option_cache_dereference: %s(%d)",
1631 #if defined (POINTER_DEBUG)
1639 rc_register (file
, line
, ptr
, *ptr
, (*ptr
) -> refcnt
, 1, RC_MISC
);
1640 if (!(*ptr
) -> refcnt
) {
1641 if ((*ptr
) -> data
.buffer
)
1642 data_string_forget (&(*ptr
) -> data
, file
, line
);
1643 if ((*ptr
) -> expression
)
1644 expression_dereference (&(*ptr
) -> expression
,
1647 option_cache_dereference (&((*ptr
) -> next
),
1649 /* Put it back on the free list... */
1650 (*ptr
) -> expression
= (struct expression
*)free_option_caches
;
1651 free_option_caches
= *ptr
;
1652 dmalloc_reuse (free_option_caches
, (char *)0, 0, 0);
1654 if ((*ptr
) -> refcnt
< 0) {
1655 log_error ("%s(%d): negative refcnt!", file
, line
);
1656 #if defined (DEBUG_RC_HISTORY)
1657 dump_rc_history (*ptr
);
1659 #if defined (POINTER_DEBUG)
1662 *ptr
= (struct option_cache
*)0;
1666 *ptr
= (struct option_cache
*)0;
1671 int hashed_option_state_dereference (universe
, state
, file
, line
)
1672 struct universe
*universe
;
1673 struct option_state
*state
;
1681 /* Get the pointer to the array of hash table bucket heads. */
1682 heads
= (pair
*)(state
-> universes
[universe
-> index
]);
1686 /* For each non-null head, loop through all the buckets dereferencing
1687 the attached option cache structures and freeing the buckets. */
1688 for (i
= 0; i
< OPTION_HASH_SIZE
; i
++) {
1689 for (cp
= heads
[i
]; cp
; cp
= next
) {
1691 option_cache_dereference
1692 ((struct option_cache
**)&cp
-> car
,
1694 free_pair (cp
, file
, line
);
1698 dfree (heads
, file
, line
);
1699 state
-> universes
[universe
-> index
] = (void *)0;
1703 int store_option (result
, universe
, packet
, lease
, client_state
,
1704 in_options
, cfg_options
, scope
, oc
)
1705 struct data_string
*result
;
1706 struct universe
*universe
;
1707 struct packet
*packet
;
1708 struct lease
*lease
;
1709 struct client_state
*client_state
;
1710 struct option_state
*in_options
;
1711 struct option_state
*cfg_options
;
1712 struct binding_scope
**scope
;
1713 struct option_cache
*oc
;
1715 struct data_string d1
, d2
;
1717 memset (&d1
, 0, sizeof d1
);
1718 memset (&d2
, 0, sizeof d2
);
1720 if (evaluate_option_cache (&d2
, packet
, lease
, client_state
,
1721 in_options
, cfg_options
, scope
, oc
, MDL
)) {
1722 if (!buffer_allocate (&d1
.buffer
,
1724 universe
-> length_size
+
1725 universe
-> tag_size
+ d2
.len
), MDL
)) {
1726 data_string_forget (result
, MDL
);
1727 data_string_forget (&d2
, MDL
);
1730 d1
.data
= &d1
.buffer
-> data
[0];
1732 memcpy (d1
.buffer
-> data
,
1733 result
-> data
, result
-> len
);
1734 d1
.len
= result
-> len
;
1735 (*universe
-> store_tag
) (&d1
.buffer
-> data
[d1
.len
],
1736 oc
-> option
-> code
);
1737 d1
.len
+= universe
-> tag_size
;
1738 (*universe
-> store_length
) (&d1
.buffer
-> data
[d1
.len
],
1740 d1
.len
+= universe
-> length_size
;
1741 memcpy (&d1
.buffer
-> data
[d1
.len
], d2
.data
, d2
.len
);
1743 data_string_forget (&d2
, MDL
);
1744 data_string_forget (result
, MDL
);
1745 data_string_copy (result
, &d1
, MDL
);
1746 data_string_forget (&d1
, MDL
);
1752 int option_space_encapsulate (result
, packet
, lease
, client_state
,
1753 in_options
, cfg_options
, scope
, name
)
1754 struct data_string
*result
;
1755 struct packet
*packet
;
1756 struct lease
*lease
;
1757 struct client_state
*client_state
;
1758 struct option_state
*in_options
;
1759 struct option_state
*cfg_options
;
1760 struct binding_scope
**scope
;
1761 struct data_string
*name
;
1765 u
= (struct universe
*)0;
1766 universe_hash_lookup (&u
, universe_hash
,
1767 (const char *)name
-> data
, name
-> len
, MDL
);
1771 if (u
-> encapsulate
)
1772 return (*u
-> encapsulate
) (result
, packet
, lease
,
1774 in_options
, cfg_options
, scope
, u
);
1775 log_error ("encapsulation requested for %s with no support.",
1780 int hashed_option_space_encapsulate (result
, packet
, lease
, client_state
,
1781 in_options
, cfg_options
, scope
, universe
)
1782 struct data_string
*result
;
1783 struct packet
*packet
;
1784 struct lease
*lease
;
1785 struct client_state
*client_state
;
1786 struct option_state
*in_options
;
1787 struct option_state
*cfg_options
;
1788 struct binding_scope
**scope
;
1789 struct universe
*universe
;
1795 if (universe
-> index
>= cfg_options
-> universe_count
)
1798 hash
= cfg_options
-> universes
[universe
-> index
];
1803 for (i
= 0; i
< OPTION_HASH_SIZE
; i
++) {
1804 for (p
= hash
[i
]; p
; p
= p
-> cdr
) {
1805 if (store_option (result
, universe
, packet
,
1806 lease
, client_state
, in_options
,
1808 (struct option_cache
*)p
-> car
))
1816 int nwip_option_space_encapsulate (result
, packet
, lease
, client_state
,
1817 in_options
, cfg_options
, scope
, universe
)
1818 struct data_string
*result
;
1819 struct packet
*packet
;
1820 struct lease
*lease
;
1821 struct client_state
*client_state
;
1822 struct option_state
*in_options
;
1823 struct option_state
*cfg_options
;
1824 struct binding_scope
**scope
;
1825 struct universe
*universe
;
1830 static struct option_cache
*no_nwip
;
1831 struct data_string ds
;
1832 struct option_chain_head
*head
;
1834 if (universe
-> index
>= cfg_options
-> universe_count
)
1836 head
= ((struct option_chain_head
*)
1837 cfg_options
-> universes
[fqdn_universe
.index
]);
1842 for (ocp
= head
-> first
; ocp
; ocp
= ocp
-> cdr
) {
1843 struct option_cache
*oc
= (struct option_cache
*)(ocp
-> car
);
1844 if (store_option (result
, universe
, packet
,
1845 lease
, client_state
, in_options
,
1847 (struct option_cache
*)ocp
-> car
))
1851 /* If there's no data, the nwip suboption is supposed to contain
1852 a suboption saying there's no data. */
1855 static unsigned char nni
[] = { 1, 0 };
1856 memset (&ds
, 0, sizeof ds
);
1859 if (option_cache_allocate (&no_nwip
, MDL
))
1860 data_string_copy (&no_nwip
-> data
, &ds
, MDL
);
1861 no_nwip
-> option
= nwip_universe
.options
[1];
1864 if (store_option (result
, universe
, packet
, lease
,
1865 client_state
, in_options
,
1866 cfg_options
, scope
, no_nwip
))
1870 memset (&ds
, 0, sizeof ds
);
1872 /* If we have nwip options, the first one has to be the
1873 nwip-exists-in-option-area option. */
1874 if (!buffer_allocate (&ds
.buffer
, result
-> len
+ 2, MDL
)) {
1875 data_string_forget (result
, MDL
);
1878 ds
.data
= &ds
.buffer
-> data
[0];
1879 ds
.buffer
-> data
[0] = 2;
1880 ds
.buffer
-> data
[1] = 0;
1881 memcpy (&ds
.buffer
-> data
[2], result
-> data
, result
-> len
);
1882 data_string_forget (result
, MDL
);
1883 data_string_copy (result
, &ds
, MDL
);
1884 data_string_forget (&ds
, MDL
);
1890 int fqdn_option_space_encapsulate (result
, packet
, lease
, client_state
,
1891 in_options
, cfg_options
, scope
, universe
)
1892 struct data_string
*result
;
1893 struct packet
*packet
;
1894 struct lease
*lease
;
1895 struct client_state
*client_state
;
1896 struct option_state
*in_options
;
1897 struct option_state
*cfg_options
;
1898 struct binding_scope
**scope
;
1899 struct universe
*universe
;
1902 struct data_string results
[FQDN_SUBOPTION_COUNT
+ 1];
1905 struct buffer
*bp
= (struct buffer
*)0;
1906 struct option_chain_head
*head
;
1908 /* If there's no FQDN universe, don't encapsulate. */
1909 if (fqdn_universe
.index
>= cfg_options
-> universe_count
)
1911 head
= ((struct option_chain_head
*)
1912 cfg_options
-> universes
[fqdn_universe
.index
]);
1916 /* Figure out the values of all the suboptions. */
1917 memset (results
, 0, sizeof results
);
1918 for (ocp
= head
-> first
; ocp
; ocp
= ocp
-> cdr
) {
1919 struct option_cache
*oc
= (struct option_cache
*)(ocp
-> car
);
1920 if (oc
-> option
-> code
> FQDN_SUBOPTION_COUNT
)
1922 evaluate_option_cache (&results
[oc
-> option
-> code
],
1923 packet
, lease
, client_state
, in_options
,
1924 cfg_options
, scope
, oc
, MDL
);
1926 len
= 4 + results
[FQDN_FQDN
].len
;
1927 /* Save the contents of the option in a buffer. */
1928 if (!buffer_allocate (&bp
, len
, MDL
)) {
1929 log_error ("no memory for option buffer.");
1932 buffer_reference (&result
-> buffer
, bp
, MDL
);
1934 result
-> data
= &bp
-> data
[0];
1936 memset (&bp
-> data
[0], 0, len
);
1937 if (results
[FQDN_NO_CLIENT_UPDATE
].len
&&
1938 results
[FQDN_NO_CLIENT_UPDATE
].data
[0])
1939 bp
-> data
[0] |= 2;
1940 if (results
[FQDN_SERVER_UPDATE
].len
&&
1941 results
[FQDN_SERVER_UPDATE
].data
[0])
1942 bp
-> data
[0] |= 1;
1943 if (results
[FQDN_RCODE1
].len
)
1944 bp
-> data
[1] = results
[FQDN_RCODE1
].data
[0];
1945 if (results
[FQDN_RCODE2
].len
)
1946 bp
-> data
[2] = results
[FQDN_RCODE2
].data
[0];
1948 if (results
[FQDN_ENCODED
].len
&&
1949 results
[FQDN_ENCODED
].data
[0]) {
1952 bp
-> data
[0] |= 4;
1953 out
= &bp
-> data
[3];
1954 if (results
[FQDN_FQDN
].len
) {
1956 while (i
< results
[FQDN_FQDN
].len
) {
1959 results
[FQDN_FQDN
].data
[j
]) &&
1960 j
< results
[FQDN_FQDN
].len
; j
++)
1963 memcpy (out
, &results
[FQDN_FQDN
].data
[i
],
1967 if (results
[FQDN_FQDN
].data
[j
] == '.')
1970 if ((results
[FQDN_FQDN
].data
1971 [results
[FQDN_FQDN
].len
- 1] == '.'))
1973 result
-> len
= out
- result
-> data
;
1974 result
-> terminated
= 0;
1977 if (results
[FQDN_FQDN
].len
) {
1978 memcpy (&bp
-> data
[3], results
[FQDN_FQDN
].data
,
1979 results
[FQDN_FQDN
].len
);
1980 result
-> len
+= results
[FQDN_FQDN
].len
;
1981 result
-> terminated
= 0;
1984 for (i
= 1; i
<= FQDN_SUBOPTION_COUNT
; i
++) {
1985 if (results
[i
].len
)
1986 data_string_forget (&results
[i
], MDL
);
1988 buffer_dereference (&bp
, MDL
);
1992 void option_space_foreach (struct packet
*packet
, struct lease
*lease
,
1993 struct client_state
*client_state
,
1994 struct option_state
*in_options
,
1995 struct option_state
*cfg_options
,
1996 struct binding_scope
**scope
,
1997 struct universe
*u
, void *stuff
,
1998 void (*func
) (struct option_cache
*,
2000 struct lease
*, struct client_state
*,
2001 struct option_state
*,
2002 struct option_state
*,
2003 struct binding_scope
**,
2004 struct universe
*, void *))
2007 (*u
-> foreach
) (packet
, lease
, client_state
, in_options
,
2008 cfg_options
, scope
, u
, stuff
, func
);
2011 void suboption_foreach (struct packet
*packet
, struct lease
*lease
,
2012 struct client_state
*client_state
,
2013 struct option_state
*in_options
,
2014 struct option_state
*cfg_options
,
2015 struct binding_scope
**scope
,
2016 struct universe
*u
, void *stuff
,
2017 void (*func
) (struct option_cache
*,
2019 struct lease
*, struct client_state
*,
2020 struct option_state
*,
2021 struct option_state
*,
2022 struct binding_scope
**,
2023 struct universe
*, void *),
2024 struct option_cache
*oc
,
2027 struct universe
*universe
= find_option_universe (oc
-> option
,
2031 if (universe
-> foreach
)
2032 (*universe
-> foreach
) (packet
, lease
, client_state
,
2033 in_options
, cfg_options
,
2034 scope
, universe
, stuff
, func
);
2037 void hashed_option_space_foreach (struct packet
*packet
, struct lease
*lease
,
2038 struct client_state
*client_state
,
2039 struct option_state
*in_options
,
2040 struct option_state
*cfg_options
,
2041 struct binding_scope
**scope
,
2042 struct universe
*u
, void *stuff
,
2043 void (*func
) (struct option_cache
*,
2046 struct client_state
*,
2047 struct option_state
*,
2048 struct option_state
*,
2049 struct binding_scope
**,
2050 struct universe
*, void *))
2054 struct option_cache
*oc
;
2056 if (cfg_options
-> universe_count
<= u
-> index
)
2059 hash
= cfg_options
-> universes
[u
-> index
];
2062 for (i
= 0; i
< OPTION_HASH_SIZE
; i
++) {
2064 /* XXX save _all_ options! XXX */
2065 for (p
= hash
[i
]; p
; p
= p
-> cdr
) {
2066 oc
= (struct option_cache
*)p
-> car
;
2067 (*func
) (oc
, packet
, lease
, client_state
,
2068 in_options
, cfg_options
, scope
, u
, stuff
);
2073 void save_linked_option (universe
, options
, oc
)
2074 struct universe
*universe
;
2075 struct option_state
*options
;
2076 struct option_cache
*oc
;
2080 struct option_chain_head
*head
;
2082 if (universe
-> index
>= options
-> universe_count
)
2084 head
= ((struct option_chain_head
*)
2085 options
-> universes
[universe
-> index
]);
2087 if (!option_chain_head_allocate (((struct option_chain_head
**)
2088 &options
-> universes
2089 [universe
-> index
]), MDL
))
2091 head
= ((struct option_chain_head
*)
2092 options
-> universes
[universe
-> index
]);
2095 /* Find the tail of the list. */
2096 for (tail
= &head
-> first
; *tail
; tail
= &((*tail
) -> cdr
)) {
2098 ((struct option_cache
*)((*tail
) -> car
)) -> option
) {
2099 option_cache_dereference ((struct option_cache
**)
2100 (&(*tail
) -> car
), MDL
);
2101 option_cache_reference ((struct option_cache
**)
2102 (&(*tail
) -> car
), oc
, MDL
);
2107 *tail
= cons (0, 0);
2109 option_cache_reference ((struct option_cache
**)
2110 (&(*tail
) -> car
), oc
, MDL
);
2114 int linked_option_space_encapsulate (result
, packet
, lease
, client_state
,
2115 in_options
, cfg_options
, scope
, universe
)
2116 struct data_string
*result
;
2117 struct packet
*packet
;
2118 struct lease
*lease
;
2119 struct client_state
*client_state
;
2120 struct option_state
*in_options
;
2121 struct option_state
*cfg_options
;
2122 struct binding_scope
**scope
;
2123 struct universe
*universe
;
2127 struct option_chain_head
*head
;
2129 if (universe
-> index
>= cfg_options
-> universe_count
)
2131 head
= ((struct option_chain_head
*)
2132 cfg_options
-> universes
[universe
-> index
]);
2137 for (oc
= head
-> first
; oc
; oc
= oc
-> cdr
) {
2138 if (store_option (result
, universe
, packet
,
2139 lease
, client_state
, in_options
, cfg_options
,
2140 scope
, (struct option_cache
*)(oc
-> car
)))
2147 void delete_linked_option (universe
, options
, code
)
2148 struct universe
*universe
;
2149 struct option_state
*options
;
2152 pair
*tail
, tmp
= (pair
)0;
2153 struct option_chain_head
*head
;
2155 if (universe
-> index
>= options
-> universe_count
)
2157 head
= ((struct option_chain_head
*)
2158 options
-> universes
[universe
-> index
]);
2162 for (tail
= &head
-> first
; *tail
; tail
= &((*tail
) -> cdr
)) {
2164 ((struct option_cache
*)(*tail
) -> car
) -> option
-> code
)
2166 tmp
= (*tail
) -> cdr
;
2167 option_cache_dereference ((struct option_cache
**)
2168 (&(*tail
) -> car
), MDL
);
2176 struct option_cache
*lookup_linked_option (universe
, options
, code
)
2177 struct universe
*universe
;
2178 struct option_state
*options
;
2182 struct option_chain_head
*head
;
2184 if (universe
-> index
>= options
-> universe_count
)
2186 head
= ((struct option_chain_head
*)
2187 options
-> universes
[universe
-> index
]);
2191 for (oc
= head
-> first
; oc
; oc
= oc
-> cdr
) {
2193 ((struct option_cache
*)(oc
-> car
)) -> option
-> code
) {
2194 return (struct option_cache
*)(oc
-> car
);
2198 return (struct option_cache
*)0;
2201 int linked_option_state_dereference (universe
, state
, file
, line
)
2202 struct universe
*universe
;
2203 struct option_state
*state
;
2207 return (option_chain_head_dereference
2208 ((struct option_chain_head
**)
2209 (&state
-> universes
[universe
-> index
]), MDL
));
2212 void linked_option_space_foreach (struct packet
*packet
, struct lease
*lease
,
2213 struct client_state
*client_state
,
2214 struct option_state
*in_options
,
2215 struct option_state
*cfg_options
,
2216 struct binding_scope
**scope
,
2217 struct universe
*u
, void *stuff
,
2218 void (*func
) (struct option_cache
*,
2221 struct client_state
*,
2222 struct option_state
*,
2223 struct option_state
*,
2224 struct binding_scope
**,
2225 struct universe
*, void *))
2228 struct option_chain_head
*head
;
2230 if (u
-> index
>= cfg_options
-> universe_count
)
2232 head
= ((struct option_chain_head
*)
2233 cfg_options
-> universes
[u
-> index
]);
2236 for (car
= head
-> first
; car
; car
= car
-> cdr
) {
2237 (*func
) ((struct option_cache
*)(car
-> car
),
2238 packet
, lease
, client_state
,
2239 in_options
, cfg_options
, scope
, u
, stuff
);
2243 void do_packet (interface
, packet
, len
, from_port
, from
, hfrom
)
2244 struct interface_info
*interface
;
2245 struct dhcp_packet
*packet
;
2247 unsigned int from_port
;
2249 struct hardware
*hfrom
;
2252 struct option_cache
*op
;
2253 struct packet
*decoded_packet
;
2254 #if defined (DEBUG_MEMORY_LEAKAGE)
2255 unsigned long previous_outstanding
= dmalloc_outstanding
;
2258 #if defined (TRACING)
2259 trace_inpacket_stash (interface
, packet
, len
, from_port
, from
, hfrom
);
2262 decoded_packet
= (struct packet
*)0;
2263 if (!packet_allocate (&decoded_packet
, MDL
)) {
2264 log_error ("do_packet: no memory for incoming packet!");
2267 decoded_packet
-> raw
= packet
;
2268 decoded_packet
-> packet_length
= len
;
2269 decoded_packet
-> client_port
= from_port
;
2270 decoded_packet
-> client_addr
= from
;
2271 interface_reference (&decoded_packet
-> interface
, interface
, MDL
);
2272 decoded_packet
-> haddr
= hfrom
;
2274 if (packet
-> hlen
> sizeof packet
-> chaddr
) {
2275 packet_dereference (&decoded_packet
, MDL
);
2276 log_info ("Discarding packet with bogus hlen.");
2280 /* If there's an option buffer, try to parse it. */
2281 if (decoded_packet
-> packet_length
>= DHCP_FIXED_NON_UDP
+ 4) {
2282 if (!parse_options (decoded_packet
)) {
2283 if (decoded_packet
-> options
)
2284 option_state_dereference
2285 (&decoded_packet
-> options
, MDL
);
2286 packet_dereference (&decoded_packet
, MDL
);
2290 if (decoded_packet
-> options_valid
&&
2291 (op
= lookup_option (&dhcp_universe
,
2292 decoded_packet
-> options
,
2293 DHO_DHCP_MESSAGE_TYPE
))) {
2294 struct data_string dp
;
2295 memset (&dp
, 0, sizeof dp
);
2296 evaluate_option_cache (&dp
, decoded_packet
,
2298 (struct client_state
*)0,
2299 decoded_packet
-> options
,
2300 (struct option_state
*)0,
2301 (struct binding_scope
**)0,
2304 decoded_packet
-> packet_type
= dp
.data
[0];
2306 decoded_packet
-> packet_type
= 0;
2307 data_string_forget (&dp
, MDL
);
2311 if (decoded_packet
-> packet_type
)
2312 dhcp (decoded_packet
);
2314 bootp (decoded_packet
);
2316 /* If the caller kept the packet, they'll have upped the refcnt. */
2317 packet_dereference (&decoded_packet
, MDL
);
2319 #if defined (DEBUG_MEMORY_LEAKAGE)
2320 log_info ("generation %ld: %ld new, %ld outstanding, %ld long-term",
2322 dmalloc_outstanding
- previous_outstanding
,
2323 dmalloc_outstanding
, dmalloc_longterm
);
2325 #if defined (DEBUG_MEMORY_LEAKAGE)
2326 dmalloc_dump_outstanding ();
2328 #if defined (DEBUG_RC_HISTORY_EXHAUSTIVELY)
2329 dump_rc_history (0);