xref: /aosp_15_r20/external/tcpdump/print-dhcp6.c (revision 05b00f6010a2396e3db2409989fc67270046269f)
1*05b00f60SXin Li /*
2*05b00f60SXin Li  * Copyright (C) 1998 and 1999 WIDE Project.
3*05b00f60SXin Li  * All rights reserved.
4*05b00f60SXin Li  *
5*05b00f60SXin Li  * Redistribution and use in source and binary forms, with or without
6*05b00f60SXin Li  * modification, are permitted provided that the following conditions
7*05b00f60SXin Li  * are met:
8*05b00f60SXin Li  * 1. Redistributions of source code must retain the above copyright
9*05b00f60SXin Li  *    notice, this list of conditions and the following disclaimer.
10*05b00f60SXin Li  * 2. Redistributions in binary form must reproduce the above copyright
11*05b00f60SXin Li  *    notice, this list of conditions and the following disclaimer in the
12*05b00f60SXin Li  *    documentation and/or other materials provided with the distribution.
13*05b00f60SXin Li  * 3. Neither the name of the project nor the names of its contributors
14*05b00f60SXin Li  *    may be used to endorse or promote products derived from this software
15*05b00f60SXin Li  *    without specific prior written permission.
16*05b00f60SXin Li  *
17*05b00f60SXin Li  * THIS SOFTWARE IS PROVIDED BY THE PROJECT AND CONTRIBUTORS ``AS IS'' AND
18*05b00f60SXin Li  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19*05b00f60SXin Li  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20*05b00f60SXin Li  * ARE DISCLAIMED.  IN NO EVENT SHALL THE PROJECT OR CONTRIBUTORS BE LIABLE
21*05b00f60SXin Li  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22*05b00f60SXin Li  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23*05b00f60SXin Li  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24*05b00f60SXin Li  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25*05b00f60SXin Li  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26*05b00f60SXin Li  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27*05b00f60SXin Li  * SUCH DAMAGE.
28*05b00f60SXin Li  */
29*05b00f60SXin Li 
30*05b00f60SXin Li /* \summary: IPv6 DHCP printer */
31*05b00f60SXin Li 
32*05b00f60SXin Li /*
33*05b00f60SXin Li  * RFC3315: DHCPv6
34*05b00f60SXin Li  * supported DHCPv6 options:
35*05b00f60SXin Li  *  RFC3319: Session Initiation Protocol (SIP) Servers options,
36*05b00f60SXin Li  *  RFC3633: IPv6 Prefix options,
37*05b00f60SXin Li  *  RFC3646: DNS Configuration options,
38*05b00f60SXin Li  *  RFC3898: Network Information Service (NIS) Configuration options,
39*05b00f60SXin Li  *  RFC4075: Simple Network Time Protocol (SNTP) Configuration option,
40*05b00f60SXin Li  *  RFC4242: Information Refresh Time option,
41*05b00f60SXin Li  *  RFC4280: Broadcast and Multicast Control Servers options,
42*05b00f60SXin Li  *  RFC5908: Network Time Protocol (NTP) Server Option for DHCPv6
43*05b00f60SXin Li  *  RFC6334: Dual-Stack Lite option,
44*05b00f60SXin Li  */
45*05b00f60SXin Li 
46*05b00f60SXin Li #ifdef HAVE_CONFIG_H
47*05b00f60SXin Li #include <config.h>
48*05b00f60SXin Li #endif
49*05b00f60SXin Li 
50*05b00f60SXin Li #include "netdissect-stdinc.h"
51*05b00f60SXin Li 
52*05b00f60SXin Li #include "netdissect.h"
53*05b00f60SXin Li #include "addrtoname.h"
54*05b00f60SXin Li #include "extract.h"
55*05b00f60SXin Li 
56*05b00f60SXin Li /* lease duration */
57*05b00f60SXin Li #define DHCP6_DURATION_INFINITE 0xffffffff
58*05b00f60SXin Li 
59*05b00f60SXin Li /* Error Values */
60*05b00f60SXin Li #define DH6ERR_FAILURE		16
61*05b00f60SXin Li #define DH6ERR_AUTHFAIL		17
62*05b00f60SXin Li #define DH6ERR_POORLYFORMED	18
63*05b00f60SXin Li #define DH6ERR_UNAVAIL		19
64*05b00f60SXin Li #define DH6ERR_OPTUNAVAIL	20
65*05b00f60SXin Li 
66*05b00f60SXin Li /* Message type */
67*05b00f60SXin Li #define DH6_SOLICIT	1
68*05b00f60SXin Li #define DH6_ADVERTISE	2
69*05b00f60SXin Li #define DH6_REQUEST	3
70*05b00f60SXin Li #define DH6_CONFIRM	4
71*05b00f60SXin Li #define DH6_RENEW	5
72*05b00f60SXin Li #define DH6_REBIND	6
73*05b00f60SXin Li #define DH6_REPLY	7
74*05b00f60SXin Li #define DH6_RELEASE	8
75*05b00f60SXin Li #define DH6_DECLINE	9
76*05b00f60SXin Li #define DH6_RECONFIGURE	10
77*05b00f60SXin Li #define DH6_INFORM_REQ	11
78*05b00f60SXin Li #define DH6_RELAY_FORW	12
79*05b00f60SXin Li #define DH6_RELAY_REPLY	13
80*05b00f60SXin Li #define DH6_LEASEQUERY	14
81*05b00f60SXin Li #define DH6_LQ_REPLY	15
82*05b00f60SXin Li 
83*05b00f60SXin Li static const struct tok dh6_msgtype_str[] = {
84*05b00f60SXin Li 	{ DH6_SOLICIT,     "solicit"          },
85*05b00f60SXin Li 	{ DH6_ADVERTISE,   "advertise"        },
86*05b00f60SXin Li 	{ DH6_REQUEST,     "request"          },
87*05b00f60SXin Li 	{ DH6_CONFIRM,     "confirm"          },
88*05b00f60SXin Li 	{ DH6_RENEW,       "renew"            },
89*05b00f60SXin Li 	{ DH6_REBIND,      "rebind"           },
90*05b00f60SXin Li 	{ DH6_REPLY,       "reply"            },
91*05b00f60SXin Li 	{ DH6_RELEASE,     "release"          },
92*05b00f60SXin Li 	{ DH6_DECLINE,     "decline"          },
93*05b00f60SXin Li 	{ DH6_RECONFIGURE, "reconfigure"      },
94*05b00f60SXin Li 	{ DH6_INFORM_REQ,  "inf-req"          },
95*05b00f60SXin Li 	{ DH6_RELAY_FORW,  "relay-fwd"        },
96*05b00f60SXin Li 	{ DH6_RELAY_REPLY, "relay-reply"      },
97*05b00f60SXin Li 	{ DH6_LEASEQUERY,  "leasequery"       },
98*05b00f60SXin Li 	{ DH6_LQ_REPLY,    "leasequery-reply" },
99*05b00f60SXin Li 	{ 0, NULL }
100*05b00f60SXin Li };
101*05b00f60SXin Li 
102*05b00f60SXin Li /* DHCP6 base packet format */
103*05b00f60SXin Li struct dhcp6 {
104*05b00f60SXin Li 	union {
105*05b00f60SXin Li 		nd_uint8_t msgtype;
106*05b00f60SXin Li 		nd_uint32_t xid;
107*05b00f60SXin Li 	} dh6_msgtypexid;
108*05b00f60SXin Li 	/* options follow */
109*05b00f60SXin Li };
110*05b00f60SXin Li #define DH6_XIDMASK	0x00ffffff
111*05b00f60SXin Li 
112*05b00f60SXin Li /* DHCPv6 relay messages */
113*05b00f60SXin Li struct dhcp6_relay {
114*05b00f60SXin Li 	nd_uint8_t dh6relay_msgtype;
115*05b00f60SXin Li 	nd_uint8_t dh6relay_hcnt;
116*05b00f60SXin Li 	nd_ipv6    dh6relay_linkaddr;	/* XXX: badly aligned */
117*05b00f60SXin Li 	nd_ipv6    dh6relay_peeraddr;
118*05b00f60SXin Li 	/* options follow */
119*05b00f60SXin Li };
120*05b00f60SXin Li 
121*05b00f60SXin Li /* options */
122*05b00f60SXin Li #define DH6OPT_CLIENTID	1
123*05b00f60SXin Li #define DH6OPT_SERVERID	2
124*05b00f60SXin Li #define DH6OPT_IA_NA 3
125*05b00f60SXin Li #define DH6OPT_IA_TA 4
126*05b00f60SXin Li #define DH6OPT_IA_ADDR 5
127*05b00f60SXin Li #define DH6OPT_ORO 6
128*05b00f60SXin Li #define DH6OPT_PREFERENCE 7
129*05b00f60SXin Li #  define DH6OPT_PREF_MAX 255
130*05b00f60SXin Li #define DH6OPT_ELAPSED_TIME 8
131*05b00f60SXin Li #define DH6OPT_RELAY_MSG 9
132*05b00f60SXin Li /*#define DH6OPT_SERVER_MSG 10 deprecated */
133*05b00f60SXin Li #define DH6OPT_AUTH 11
134*05b00f60SXin Li #  define DH6OPT_AUTHPROTO_DELAYED 2
135*05b00f60SXin Li #  define DH6OPT_AUTHPROTO_RECONFIG 3
136*05b00f60SXin Li #  define DH6OPT_AUTHALG_HMACMD5 1
137*05b00f60SXin Li #  define DH6OPT_AUTHRDM_MONOCOUNTER 0
138*05b00f60SXin Li #  define DH6OPT_AUTHRECONFIG_KEY 1
139*05b00f60SXin Li #  define DH6OPT_AUTHRECONFIG_HMACMD5 2
140*05b00f60SXin Li #define DH6OPT_UNICAST 12
141*05b00f60SXin Li #define DH6OPT_STATUS_CODE 13
142*05b00f60SXin Li #  define DH6OPT_STCODE_SUCCESS 0
143*05b00f60SXin Li #  define DH6OPT_STCODE_UNSPECFAIL 1
144*05b00f60SXin Li #  define DH6OPT_STCODE_NOADDRAVAIL 2
145*05b00f60SXin Li #  define DH6OPT_STCODE_NOBINDING 3
146*05b00f60SXin Li #  define DH6OPT_STCODE_NOTONLINK 4
147*05b00f60SXin Li #  define DH6OPT_STCODE_USEMULTICAST 5
148*05b00f60SXin Li #  define DH6OPT_STCODE_NOPREFIXAVAIL 6
149*05b00f60SXin Li #  define DH6OPT_STCODE_UNKNOWNQUERYTYPE 7
150*05b00f60SXin Li #  define DH6OPT_STCODE_MALFORMEDQUERY 8
151*05b00f60SXin Li #  define DH6OPT_STCODE_NOTCONFIGURED 9
152*05b00f60SXin Li #  define DH6OPT_STCODE_NOTALLOWED 10
153*05b00f60SXin Li #define DH6OPT_RAPID_COMMIT 14
154*05b00f60SXin Li #define DH6OPT_USER_CLASS 15
155*05b00f60SXin Li #define DH6OPT_VENDOR_CLASS 16
156*05b00f60SXin Li #define DH6OPT_VENDOR_OPTS 17
157*05b00f60SXin Li #define DH6OPT_INTERFACE_ID 18
158*05b00f60SXin Li #define DH6OPT_RECONF_MSG 19
159*05b00f60SXin Li #define DH6OPT_RECONF_ACCEPT 20
160*05b00f60SXin Li #define DH6OPT_SIP_SERVER_D 21
161*05b00f60SXin Li #define DH6OPT_SIP_SERVER_A 22
162*05b00f60SXin Li #define DH6OPT_DNS_SERVERS 23
163*05b00f60SXin Li #define DH6OPT_DOMAIN_LIST 24
164*05b00f60SXin Li #define DH6OPT_IA_PD 25
165*05b00f60SXin Li #define DH6OPT_IA_PD_PREFIX 26
166*05b00f60SXin Li #define DH6OPT_NIS_SERVERS 27
167*05b00f60SXin Li #define DH6OPT_NISP_SERVERS 28
168*05b00f60SXin Li #define DH6OPT_NIS_NAME 29
169*05b00f60SXin Li #define DH6OPT_NISP_NAME 30
170*05b00f60SXin Li #define DH6OPT_SNTP_SERVERS 31
171*05b00f60SXin Li #define DH6OPT_LIFETIME 32
172*05b00f60SXin Li #define DH6OPT_BCMCS_SERVER_D 33
173*05b00f60SXin Li #define DH6OPT_BCMCS_SERVER_A 34
174*05b00f60SXin Li #define DH6OPT_GEOCONF_CIVIC 36
175*05b00f60SXin Li #define DH6OPT_REMOTE_ID 37
176*05b00f60SXin Li #define DH6OPT_SUBSCRIBER_ID 38
177*05b00f60SXin Li #define DH6OPT_CLIENT_FQDN 39
178*05b00f60SXin Li #define DH6OPT_PANA_AGENT 40
179*05b00f60SXin Li #define DH6OPT_NEW_POSIX_TIMEZONE 41
180*05b00f60SXin Li #define DH6OPT_NEW_TZDB_TIMEZONE 42
181*05b00f60SXin Li #define DH6OPT_ERO 43
182*05b00f60SXin Li #define DH6OPT_LQ_QUERY 44
183*05b00f60SXin Li #define DH6OPT_CLIENT_DATA 45
184*05b00f60SXin Li #define DH6OPT_CLT_TIME 46
185*05b00f60SXin Li #define DH6OPT_LQ_RELAY_DATA 47
186*05b00f60SXin Li #define DH6OPT_LQ_CLIENT_LINK 48
187*05b00f60SXin Li #define DH6OPT_NTP_SERVER 56
188*05b00f60SXin Li #  define DH6OPT_NTP_SUBOPTION_SRV_ADDR 1
189*05b00f60SXin Li #  define DH6OPT_NTP_SUBOPTION_MC_ADDR 2
190*05b00f60SXin Li #  define DH6OPT_NTP_SUBOPTION_SRV_FQDN 3
191*05b00f60SXin Li #define DH6OPT_AFTR_NAME 64
192*05b00f60SXin Li #define DH6OPT_MUDURL 112
193*05b00f60SXin Li 
194*05b00f60SXin Li static const struct tok dh6opt_str[] = {
195*05b00f60SXin Li 	{ DH6OPT_CLIENTID,           "client-ID"            },
196*05b00f60SXin Li 	{ DH6OPT_SERVERID,           "server-ID"            },
197*05b00f60SXin Li 	{ DH6OPT_IA_NA,              "IA_NA"                },
198*05b00f60SXin Li 	{ DH6OPT_IA_TA,              "IA_TA"                },
199*05b00f60SXin Li 	{ DH6OPT_IA_ADDR,            "IA_ADDR"              },
200*05b00f60SXin Li 	{ DH6OPT_ORO,                "option-request"       },
201*05b00f60SXin Li 	{ DH6OPT_PREFERENCE,         "preference"           },
202*05b00f60SXin Li 	{ DH6OPT_ELAPSED_TIME,       "elapsed-time"         },
203*05b00f60SXin Li 	{ DH6OPT_RELAY_MSG,          "relay-message"        },
204*05b00f60SXin Li 	{ DH6OPT_AUTH,               "authentication"       },
205*05b00f60SXin Li 	{ DH6OPT_UNICAST,            "server-unicast"       },
206*05b00f60SXin Li 	{ DH6OPT_STATUS_CODE,        "status-code"          },
207*05b00f60SXin Li 	{ DH6OPT_RAPID_COMMIT,       "rapid-commit"         },
208*05b00f60SXin Li 	{ DH6OPT_USER_CLASS,         "user-class"           },
209*05b00f60SXin Li 	{ DH6OPT_VENDOR_CLASS,       "vendor-class"         },
210*05b00f60SXin Li 	{ DH6OPT_VENDOR_OPTS,        "vendor-specific-info" },
211*05b00f60SXin Li 	{ DH6OPT_INTERFACE_ID,       "interface-ID"         },
212*05b00f60SXin Li 	{ DH6OPT_RECONF_MSG,         "reconfigure-message"  },
213*05b00f60SXin Li 	{ DH6OPT_RECONF_ACCEPT,      "reconfigure-accept"   },
214*05b00f60SXin Li 	{ DH6OPT_SIP_SERVER_D,       "SIP-servers-domain"   },
215*05b00f60SXin Li 	{ DH6OPT_SIP_SERVER_A,       "SIP-servers-address"  },
216*05b00f60SXin Li 	{ DH6OPT_DNS_SERVERS,        "DNS-server"           },
217*05b00f60SXin Li 	{ DH6OPT_DOMAIN_LIST,        "DNS-search-list"      },
218*05b00f60SXin Li 	{ DH6OPT_IA_PD,              "IA_PD"                },
219*05b00f60SXin Li 	{ DH6OPT_IA_PD_PREFIX,       "IA_PD-prefix"         },
220*05b00f60SXin Li 	{ DH6OPT_SNTP_SERVERS,       "SNTP-servers"         },
221*05b00f60SXin Li 	{ DH6OPT_LIFETIME,           "lifetime"             },
222*05b00f60SXin Li 	{ DH6OPT_NIS_SERVERS,        "NIS-server"           },
223*05b00f60SXin Li 	{ DH6OPT_NISP_SERVERS,       "NIS+-server"          },
224*05b00f60SXin Li 	{ DH6OPT_NIS_NAME,           "NIS-domain-name"      },
225*05b00f60SXin Li 	{ DH6OPT_NISP_NAME,          "NIS+-domain-name"     },
226*05b00f60SXin Li 	{ DH6OPT_BCMCS_SERVER_D,     "BCMCS-domain-name"    },
227*05b00f60SXin Li 	{ DH6OPT_BCMCS_SERVER_A,     "BCMCS-server"         },
228*05b00f60SXin Li 	{ DH6OPT_GEOCONF_CIVIC,      "Geoconf-Civic"        },
229*05b00f60SXin Li 	{ DH6OPT_REMOTE_ID,          "Remote-ID"            },
230*05b00f60SXin Li 	{ DH6OPT_SUBSCRIBER_ID,      "Subscriber-ID"        },
231*05b00f60SXin Li 	{ DH6OPT_CLIENT_FQDN,        "Client-FQDN"          },
232*05b00f60SXin Li 	{ DH6OPT_PANA_AGENT,         "PANA-agent"           },
233*05b00f60SXin Li 	{ DH6OPT_NEW_POSIX_TIMEZONE, "POSIX-timezone"       },
234*05b00f60SXin Li 	{ DH6OPT_NEW_TZDB_TIMEZONE,  "POSIX-tz-database"    },
235*05b00f60SXin Li 	{ DH6OPT_ERO,                "Echo-request-option"  },
236*05b00f60SXin Li 	{ DH6OPT_LQ_QUERY,           "Lease-query"          },
237*05b00f60SXin Li 	{ DH6OPT_CLIENT_DATA,        "LQ-client-data"       },
238*05b00f60SXin Li 	{ DH6OPT_CLT_TIME,           "Clt-time"             },
239*05b00f60SXin Li 	{ DH6OPT_LQ_RELAY_DATA,      "LQ-relay-data"        },
240*05b00f60SXin Li 	{ DH6OPT_LQ_CLIENT_LINK,     "LQ-client-link"       },
241*05b00f60SXin Li 	{ DH6OPT_NTP_SERVER,         "NTP-server"           },
242*05b00f60SXin Li 	{ DH6OPT_AFTR_NAME,          "AFTR-Name"            },
243*05b00f60SXin Li 	{ DH6OPT_MUDURL,             "MUD-URL"              },
244*05b00f60SXin Li 	{ 0, NULL }
245*05b00f60SXin Li };
246*05b00f60SXin Li 
247*05b00f60SXin Li static const struct tok dh6opt_stcode_str[] = {
248*05b00f60SXin Li 	{ DH6OPT_STCODE_SUCCESS,          "Success"          }, /* RFC3315 */
249*05b00f60SXin Li 	{ DH6OPT_STCODE_UNSPECFAIL,       "UnspecFail"       }, /* RFC3315 */
250*05b00f60SXin Li 	{ DH6OPT_STCODE_NOADDRAVAIL,      "NoAddrsAvail"     }, /* RFC3315 */
251*05b00f60SXin Li 	{ DH6OPT_STCODE_NOBINDING,        "NoBinding"        }, /* RFC3315 */
252*05b00f60SXin Li 	{ DH6OPT_STCODE_NOTONLINK,        "NotOnLink"        }, /* RFC3315 */
253*05b00f60SXin Li 	{ DH6OPT_STCODE_USEMULTICAST,     "UseMulticast"     }, /* RFC3315 */
254*05b00f60SXin Li 	{ DH6OPT_STCODE_NOPREFIXAVAIL,    "NoPrefixAvail"    }, /* RFC3633 */
255*05b00f60SXin Li 	{ DH6OPT_STCODE_UNKNOWNQUERYTYPE, "UnknownQueryType" }, /* RFC5007 */
256*05b00f60SXin Li 	{ DH6OPT_STCODE_MALFORMEDQUERY,   "MalformedQuery"   }, /* RFC5007 */
257*05b00f60SXin Li 	{ DH6OPT_STCODE_NOTCONFIGURED,    "NotConfigured"    }, /* RFC5007 */
258*05b00f60SXin Li 	{ DH6OPT_STCODE_NOTALLOWED,       "NotAllowed"       }, /* RFC5007 */
259*05b00f60SXin Li 	{ 0, NULL }
260*05b00f60SXin Li };
261*05b00f60SXin Li 
262*05b00f60SXin Li struct dhcp6opt {
263*05b00f60SXin Li 	nd_uint16_t dh6opt_type;
264*05b00f60SXin Li 	nd_uint16_t dh6opt_len;
265*05b00f60SXin Li 	/* type-dependent data follows */
266*05b00f60SXin Li };
267*05b00f60SXin Li 
268*05b00f60SXin Li static const char *
dhcp6stcode(const uint16_t code)269*05b00f60SXin Li dhcp6stcode(const uint16_t code)
270*05b00f60SXin Li {
271*05b00f60SXin Li 	return code > 255 ? "INVALID code" : tok2str(dh6opt_stcode_str, "code%u", code);
272*05b00f60SXin Li }
273*05b00f60SXin Li 
274*05b00f60SXin Li static void
dhcp6opt_print(netdissect_options * ndo,const u_char * cp,const u_char * ep)275*05b00f60SXin Li dhcp6opt_print(netdissect_options *ndo,
276*05b00f60SXin Li                const u_char *cp, const u_char *ep)
277*05b00f60SXin Li {
278*05b00f60SXin Li 	const struct dhcp6opt *dh6o;
279*05b00f60SXin Li 	const u_char *tp;
280*05b00f60SXin Li 	u_int i;
281*05b00f60SXin Li 	uint16_t opttype;
282*05b00f60SXin Li 	uint16_t optlen;
283*05b00f60SXin Li 	uint8_t auth_proto;
284*05b00f60SXin Li 	uint8_t auth_alg;
285*05b00f60SXin Li 	uint8_t auth_rdm;
286*05b00f60SXin Li 	u_int authinfolen, authrealmlen;
287*05b00f60SXin Li 	u_int remain_len;  /* Length of remaining options */
288*05b00f60SXin Li 	u_int label_len;   /* Label length */
289*05b00f60SXin Li 	uint16_t subopt_code;
290*05b00f60SXin Li 	uint16_t subopt_len;
291*05b00f60SXin Li 	uint8_t dh6_reconf_type;
292*05b00f60SXin Li 	uint8_t dh6_lq_query_type;
293*05b00f60SXin Li 
294*05b00f60SXin Li 	if (cp == ep)
295*05b00f60SXin Li 		return;
296*05b00f60SXin Li 	while (cp < ep) {
297*05b00f60SXin Li 		if (ep < cp + sizeof(*dh6o))
298*05b00f60SXin Li 			goto trunc;
299*05b00f60SXin Li 		dh6o = (const struct dhcp6opt *)cp;
300*05b00f60SXin Li 		ND_TCHECK_SIZE(dh6o);
301*05b00f60SXin Li 		optlen = GET_BE_U_2(dh6o->dh6opt_len);
302*05b00f60SXin Li 		if (ep < cp + sizeof(*dh6o) + optlen)
303*05b00f60SXin Li 			goto trunc;
304*05b00f60SXin Li 		opttype = GET_BE_U_2(dh6o->dh6opt_type);
305*05b00f60SXin Li 		ND_PRINT(" (%s", tok2str(dh6opt_str, "opt_%u", opttype));
306*05b00f60SXin Li 		ND_TCHECK_LEN(cp + sizeof(*dh6o), optlen);
307*05b00f60SXin Li 		switch (opttype) {
308*05b00f60SXin Li 		case DH6OPT_CLIENTID:
309*05b00f60SXin Li 		case DH6OPT_SERVERID:
310*05b00f60SXin Li 			if (optlen < 2) {
311*05b00f60SXin Li 				/*(*/
312*05b00f60SXin Li 				ND_PRINT(" ?)");
313*05b00f60SXin Li 				break;
314*05b00f60SXin Li 			}
315*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
316*05b00f60SXin Li 			switch (GET_BE_U_2(tp)) {
317*05b00f60SXin Li 			case 1:
318*05b00f60SXin Li 				if (optlen >= 2 + 6) {
319*05b00f60SXin Li 					ND_PRINT(" hwaddr/time type %u time %u ",
320*05b00f60SXin Li 					    GET_BE_U_2(tp + 2),
321*05b00f60SXin Li 					    GET_BE_U_4(tp + 4));
322*05b00f60SXin Li 					for (i = 8; i < optlen; i++)
323*05b00f60SXin Li 						ND_PRINT("%02x",
324*05b00f60SXin Li 							 GET_U_1(tp + i));
325*05b00f60SXin Li 					/*(*/
326*05b00f60SXin Li 					ND_PRINT(")");
327*05b00f60SXin Li 				} else {
328*05b00f60SXin Li 					/*(*/
329*05b00f60SXin Li 					ND_PRINT(" ?)");
330*05b00f60SXin Li 				}
331*05b00f60SXin Li 				break;
332*05b00f60SXin Li 			case 2:
333*05b00f60SXin Li 				if (optlen >= 2 + 8) {
334*05b00f60SXin Li 					ND_PRINT(" vid ");
335*05b00f60SXin Li 					for (i = 2; i < 2 + 8; i++)
336*05b00f60SXin Li 						ND_PRINT("%02x",
337*05b00f60SXin Li 							 GET_U_1(tp + i));
338*05b00f60SXin Li 					/*(*/
339*05b00f60SXin Li 					ND_PRINT(")");
340*05b00f60SXin Li 				} else {
341*05b00f60SXin Li 					/*(*/
342*05b00f60SXin Li 					ND_PRINT(" ?)");
343*05b00f60SXin Li 				}
344*05b00f60SXin Li 				break;
345*05b00f60SXin Li 			case 3:
346*05b00f60SXin Li 				if (optlen >= 2 + 2) {
347*05b00f60SXin Li 					ND_PRINT(" hwaddr type %u ",
348*05b00f60SXin Li 					    GET_BE_U_2(tp + 2));
349*05b00f60SXin Li 					for (i = 4; i < optlen; i++)
350*05b00f60SXin Li 						ND_PRINT("%02x",
351*05b00f60SXin Li 							 GET_U_1(tp + i));
352*05b00f60SXin Li 					/*(*/
353*05b00f60SXin Li 					ND_PRINT(")");
354*05b00f60SXin Li 				} else {
355*05b00f60SXin Li 					/*(*/
356*05b00f60SXin Li 					ND_PRINT(" ?)");
357*05b00f60SXin Li 				}
358*05b00f60SXin Li 				break;
359*05b00f60SXin Li 			default:
360*05b00f60SXin Li 				ND_PRINT(" type %u)", GET_BE_U_2(tp));
361*05b00f60SXin Li 				break;
362*05b00f60SXin Li 			}
363*05b00f60SXin Li 			break;
364*05b00f60SXin Li 		case DH6OPT_IA_ADDR:
365*05b00f60SXin Li 			if (optlen < 24) {
366*05b00f60SXin Li 				/*(*/
367*05b00f60SXin Li 				ND_PRINT(" ?)");
368*05b00f60SXin Li 				break;
369*05b00f60SXin Li 			}
370*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
371*05b00f60SXin Li 			ND_PRINT(" %s", GET_IP6ADDR_STRING(tp));
372*05b00f60SXin Li 			ND_PRINT(" pltime:%u vltime:%u",
373*05b00f60SXin Li 			    GET_BE_U_4(tp + 16),
374*05b00f60SXin Li 			    GET_BE_U_4(tp + 20));
375*05b00f60SXin Li 			if (optlen > 24) {
376*05b00f60SXin Li 				/* there are sub-options */
377*05b00f60SXin Li 				dhcp6opt_print(ndo, tp + 24, tp + optlen);
378*05b00f60SXin Li 			}
379*05b00f60SXin Li 			ND_PRINT(")");
380*05b00f60SXin Li 			break;
381*05b00f60SXin Li 		case DH6OPT_ORO:
382*05b00f60SXin Li 		case DH6OPT_ERO:
383*05b00f60SXin Li 			if (optlen % 2) {
384*05b00f60SXin Li 				ND_PRINT(" ?)");
385*05b00f60SXin Li 				break;
386*05b00f60SXin Li 			}
387*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
388*05b00f60SXin Li 			for (i = 0; i < optlen; i += 2) {
389*05b00f60SXin Li 				ND_PRINT(" %s",
390*05b00f60SXin Li 				    tok2str(dh6opt_str, "opt_%u", GET_BE_U_2(tp + i)));
391*05b00f60SXin Li 			}
392*05b00f60SXin Li 			ND_PRINT(")");
393*05b00f60SXin Li 			break;
394*05b00f60SXin Li 		case DH6OPT_PREFERENCE:
395*05b00f60SXin Li 			if (optlen != 1) {
396*05b00f60SXin Li 				ND_PRINT(" ?)");
397*05b00f60SXin Li 				break;
398*05b00f60SXin Li 			}
399*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
400*05b00f60SXin Li 			ND_PRINT(" %u)", GET_U_1(tp));
401*05b00f60SXin Li 			break;
402*05b00f60SXin Li 		case DH6OPT_ELAPSED_TIME:
403*05b00f60SXin Li 			if (optlen != 2) {
404*05b00f60SXin Li 				ND_PRINT(" ?)");
405*05b00f60SXin Li 				break;
406*05b00f60SXin Li 			}
407*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
408*05b00f60SXin Li 			ND_PRINT(" %u)", GET_BE_U_2(tp));
409*05b00f60SXin Li 			break;
410*05b00f60SXin Li 		case DH6OPT_RELAY_MSG:
411*05b00f60SXin Li 		    {
412*05b00f60SXin Li 			const u_char *snapend_save;
413*05b00f60SXin Li 
414*05b00f60SXin Li 			ND_PRINT(" (");
415*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
416*05b00f60SXin Li 			/*
417*05b00f60SXin Li 			 * Update the snapend to the end of the option before
418*05b00f60SXin Li 			 * calling recursively dhcp6_print() for the nested
419*05b00f60SXin Li 			 * packet. Other options may be present after the
420*05b00f60SXin Li 			 * nested DHCPv6 packet. This prevents that, in
421*05b00f60SXin Li 			 * dhcp6_print(), for the nested DHCPv6 packet, the
422*05b00f60SXin Li 			 * remaining length < remaining caplen.
423*05b00f60SXin Li 			 */
424*05b00f60SXin Li 			snapend_save = ndo->ndo_snapend;
425*05b00f60SXin Li 			ndo->ndo_snapend = ND_MIN(tp + optlen, ndo->ndo_snapend);
426*05b00f60SXin Li 			dhcp6_print(ndo, tp, optlen);
427*05b00f60SXin Li 			ndo->ndo_snapend = snapend_save;
428*05b00f60SXin Li 			ND_PRINT(")");
429*05b00f60SXin Li 			break;
430*05b00f60SXin Li 		    }
431*05b00f60SXin Li 		case DH6OPT_AUTH:
432*05b00f60SXin Li 			if (optlen < 11) {
433*05b00f60SXin Li 				ND_PRINT(" ?)");
434*05b00f60SXin Li 				break;
435*05b00f60SXin Li 			}
436*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
437*05b00f60SXin Li 			auth_proto = GET_U_1(tp);
438*05b00f60SXin Li 			switch (auth_proto) {
439*05b00f60SXin Li 			case DH6OPT_AUTHPROTO_DELAYED:
440*05b00f60SXin Li 				ND_PRINT(" proto: delayed");
441*05b00f60SXin Li 				break;
442*05b00f60SXin Li 			case DH6OPT_AUTHPROTO_RECONFIG:
443*05b00f60SXin Li 				ND_PRINT(" proto: reconfigure");
444*05b00f60SXin Li 				break;
445*05b00f60SXin Li 			default:
446*05b00f60SXin Li 				ND_PRINT(" proto: %u", auth_proto);
447*05b00f60SXin Li 				break;
448*05b00f60SXin Li 			}
449*05b00f60SXin Li 			tp++;
450*05b00f60SXin Li 			auth_alg = GET_U_1(tp);
451*05b00f60SXin Li 			switch (auth_alg) {
452*05b00f60SXin Li 			case DH6OPT_AUTHALG_HMACMD5:
453*05b00f60SXin Li 				/* XXX: may depend on the protocol */
454*05b00f60SXin Li 				ND_PRINT(", alg: HMAC-MD5");
455*05b00f60SXin Li 				break;
456*05b00f60SXin Li 			default:
457*05b00f60SXin Li 				ND_PRINT(", alg: %u", auth_alg);
458*05b00f60SXin Li 				break;
459*05b00f60SXin Li 			}
460*05b00f60SXin Li 			tp++;
461*05b00f60SXin Li 			auth_rdm = GET_U_1(tp);
462*05b00f60SXin Li 			switch (auth_rdm) {
463*05b00f60SXin Li 			case DH6OPT_AUTHRDM_MONOCOUNTER:
464*05b00f60SXin Li 				ND_PRINT(", RDM: mono");
465*05b00f60SXin Li 				break;
466*05b00f60SXin Li 			default:
467*05b00f60SXin Li 				ND_PRINT(", RDM: %u", auth_rdm);
468*05b00f60SXin Li 				break;
469*05b00f60SXin Li 			}
470*05b00f60SXin Li 			tp++;
471*05b00f60SXin Li 			ND_PRINT(", RD:");
472*05b00f60SXin Li 			for (i = 0; i < 4; i++, tp += 2)
473*05b00f60SXin Li 				ND_PRINT(" %04x", GET_BE_U_2(tp));
474*05b00f60SXin Li 
475*05b00f60SXin Li 			/* protocol dependent part */
476*05b00f60SXin Li 			authinfolen = optlen - 11;
477*05b00f60SXin Li 			switch (auth_proto) {
478*05b00f60SXin Li 			case DH6OPT_AUTHPROTO_DELAYED:
479*05b00f60SXin Li 				if (authinfolen == 0)
480*05b00f60SXin Li 					break;
481*05b00f60SXin Li 				if (authinfolen < 20) {
482*05b00f60SXin Li 					ND_PRINT(" ??");
483*05b00f60SXin Li 					break;
484*05b00f60SXin Li 				}
485*05b00f60SXin Li 				authrealmlen = authinfolen - 20;
486*05b00f60SXin Li 				if (authrealmlen > 0) {
487*05b00f60SXin Li 					ND_PRINT(", realm: ");
488*05b00f60SXin Li 				}
489*05b00f60SXin Li 				for (i = 0; i < authrealmlen; i++, tp++)
490*05b00f60SXin Li 					ND_PRINT("%02x", GET_U_1(tp));
491*05b00f60SXin Li 				ND_PRINT(", key ID: %08x", GET_BE_U_4(tp));
492*05b00f60SXin Li 				tp += 4;
493*05b00f60SXin Li 				ND_PRINT(", HMAC-MD5:");
494*05b00f60SXin Li 				for (i = 0; i < 4; i++, tp+= 4)
495*05b00f60SXin Li 					ND_PRINT(" %08x", GET_BE_U_4(tp));
496*05b00f60SXin Li 				break;
497*05b00f60SXin Li 			case DH6OPT_AUTHPROTO_RECONFIG:
498*05b00f60SXin Li 				if (authinfolen != 17) {
499*05b00f60SXin Li 					ND_PRINT(" ??");
500*05b00f60SXin Li 					break;
501*05b00f60SXin Li 				}
502*05b00f60SXin Li 				switch (GET_U_1(tp)) {
503*05b00f60SXin Li 				case DH6OPT_AUTHRECONFIG_KEY:
504*05b00f60SXin Li 					ND_PRINT(" reconfig-key");
505*05b00f60SXin Li 					break;
506*05b00f60SXin Li 				case DH6OPT_AUTHRECONFIG_HMACMD5:
507*05b00f60SXin Li 					ND_PRINT(" type: HMAC-MD5");
508*05b00f60SXin Li 					break;
509*05b00f60SXin Li 				default:
510*05b00f60SXin Li 					ND_PRINT(" type: ??");
511*05b00f60SXin Li 					break;
512*05b00f60SXin Li 				}
513*05b00f60SXin Li 				tp++;
514*05b00f60SXin Li 				ND_PRINT(" value:");
515*05b00f60SXin Li 				for (i = 0; i < 4; i++, tp+= 4)
516*05b00f60SXin Li 					ND_PRINT(" %08x", GET_BE_U_4(tp));
517*05b00f60SXin Li 				break;
518*05b00f60SXin Li 			default:
519*05b00f60SXin Li 				ND_PRINT(" ??");
520*05b00f60SXin Li 				break;
521*05b00f60SXin Li 			}
522*05b00f60SXin Li 
523*05b00f60SXin Li 			ND_PRINT(")");
524*05b00f60SXin Li 			break;
525*05b00f60SXin Li 		case DH6OPT_RAPID_COMMIT: /* nothing todo */
526*05b00f60SXin Li 			ND_PRINT(")");
527*05b00f60SXin Li 			break;
528*05b00f60SXin Li 		case DH6OPT_INTERFACE_ID:
529*05b00f60SXin Li 		case DH6OPT_SUBSCRIBER_ID:
530*05b00f60SXin Li 			/*
531*05b00f60SXin Li 			 * Since we cannot predict the encoding, print hex dump
532*05b00f60SXin Li 			 * at most 10 characters.
533*05b00f60SXin Li 			 */
534*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
535*05b00f60SXin Li 			ND_PRINT(" ");
536*05b00f60SXin Li 			for (i = 0; i < optlen && i < 10; i++)
537*05b00f60SXin Li 				ND_PRINT("%02x", GET_U_1(tp + i));
538*05b00f60SXin Li 			ND_PRINT("...)");
539*05b00f60SXin Li 			break;
540*05b00f60SXin Li 		case DH6OPT_RECONF_MSG:
541*05b00f60SXin Li 			if (optlen != 1) {
542*05b00f60SXin Li 				ND_PRINT(" ?)");
543*05b00f60SXin Li 				break;
544*05b00f60SXin Li 			}
545*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
546*05b00f60SXin Li 			dh6_reconf_type = GET_U_1(tp);
547*05b00f60SXin Li 			switch (dh6_reconf_type) {
548*05b00f60SXin Li 			case DH6_RENEW:
549*05b00f60SXin Li 				ND_PRINT(" for renew)");
550*05b00f60SXin Li 				break;
551*05b00f60SXin Li 			case DH6_INFORM_REQ:
552*05b00f60SXin Li 				ND_PRINT(" for inf-req)");
553*05b00f60SXin Li 				break;
554*05b00f60SXin Li 			default:
555*05b00f60SXin Li 				ND_PRINT(" for ?\?\?(%02x))", dh6_reconf_type);
556*05b00f60SXin Li 				break;
557*05b00f60SXin Li 			}
558*05b00f60SXin Li 			break;
559*05b00f60SXin Li 		case DH6OPT_RECONF_ACCEPT: /* nothing todo */
560*05b00f60SXin Li 			ND_PRINT(")");
561*05b00f60SXin Li 			break;
562*05b00f60SXin Li 		case DH6OPT_SIP_SERVER_A:
563*05b00f60SXin Li 		case DH6OPT_DNS_SERVERS:
564*05b00f60SXin Li 		case DH6OPT_SNTP_SERVERS:
565*05b00f60SXin Li 		case DH6OPT_NIS_SERVERS:
566*05b00f60SXin Li 		case DH6OPT_NISP_SERVERS:
567*05b00f60SXin Li 		case DH6OPT_BCMCS_SERVER_A:
568*05b00f60SXin Li 		case DH6OPT_PANA_AGENT:
569*05b00f60SXin Li 		case DH6OPT_LQ_CLIENT_LINK:
570*05b00f60SXin Li 			if (optlen % 16) {
571*05b00f60SXin Li 				ND_PRINT(" ?)");
572*05b00f60SXin Li 				break;
573*05b00f60SXin Li 			}
574*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
575*05b00f60SXin Li 			for (i = 0; i < optlen; i += 16)
576*05b00f60SXin Li 				ND_PRINT(" %s", GET_IP6ADDR_STRING(tp + i));
577*05b00f60SXin Li 			ND_PRINT(")");
578*05b00f60SXin Li 			break;
579*05b00f60SXin Li 		case DH6OPT_SIP_SERVER_D:
580*05b00f60SXin Li 		case DH6OPT_DOMAIN_LIST:
581*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
582*05b00f60SXin Li 			while (tp < cp + sizeof(*dh6o) + optlen) {
583*05b00f60SXin Li 				ND_PRINT(" ");
584*05b00f60SXin Li 				if ((tp = fqdn_print(ndo, tp, cp + sizeof(*dh6o) + optlen)) == NULL)
585*05b00f60SXin Li 					goto trunc;
586*05b00f60SXin Li 			}
587*05b00f60SXin Li 			ND_PRINT(")");
588*05b00f60SXin Li 			break;
589*05b00f60SXin Li 		case DH6OPT_STATUS_CODE:
590*05b00f60SXin Li 			if (optlen < 2) {
591*05b00f60SXin Li 				ND_PRINT(" ?)");
592*05b00f60SXin Li 				break;
593*05b00f60SXin Li 			}
594*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
595*05b00f60SXin Li 			ND_PRINT(" %s)", dhcp6stcode(GET_BE_U_2(tp)));
596*05b00f60SXin Li 			break;
597*05b00f60SXin Li 		case DH6OPT_IA_NA:
598*05b00f60SXin Li 		case DH6OPT_IA_PD:
599*05b00f60SXin Li 			if (optlen < 12) {
600*05b00f60SXin Li 				ND_PRINT(" ?)");
601*05b00f60SXin Li 				break;
602*05b00f60SXin Li 			}
603*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
604*05b00f60SXin Li 			ND_PRINT(" IAID:%u T1:%u T2:%u",
605*05b00f60SXin Li 			    GET_BE_U_4(tp),
606*05b00f60SXin Li 			    GET_BE_U_4(tp + 4),
607*05b00f60SXin Li 			    GET_BE_U_4(tp + 8));
608*05b00f60SXin Li 			if (optlen > 12) {
609*05b00f60SXin Li 				/* there are sub-options */
610*05b00f60SXin Li 				dhcp6opt_print(ndo, tp + 12, tp + optlen);
611*05b00f60SXin Li 			}
612*05b00f60SXin Li 			ND_PRINT(")");
613*05b00f60SXin Li 			break;
614*05b00f60SXin Li 		case DH6OPT_IA_TA:
615*05b00f60SXin Li 			if (optlen < 4) {
616*05b00f60SXin Li 				ND_PRINT(" ?)");
617*05b00f60SXin Li 				break;
618*05b00f60SXin Li 			}
619*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
620*05b00f60SXin Li 			ND_PRINT(" IAID:%u", GET_BE_U_4(tp));
621*05b00f60SXin Li 			if (optlen > 4) {
622*05b00f60SXin Li 				/* there are sub-options */
623*05b00f60SXin Li 				dhcp6opt_print(ndo, tp + 4, tp + optlen);
624*05b00f60SXin Li 			}
625*05b00f60SXin Li 			ND_PRINT(")");
626*05b00f60SXin Li 			break;
627*05b00f60SXin Li 		case DH6OPT_IA_PD_PREFIX:
628*05b00f60SXin Li 			if (optlen < 25) {
629*05b00f60SXin Li 				ND_PRINT(" ?)");
630*05b00f60SXin Li 				break;
631*05b00f60SXin Li 			}
632*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
633*05b00f60SXin Li 			ND_PRINT(" %s/%u", GET_IP6ADDR_STRING(tp + 9),
634*05b00f60SXin Li 				 GET_U_1(tp + 8));
635*05b00f60SXin Li 			ND_PRINT(" pltime:%u vltime:%u",
636*05b00f60SXin Li 			    GET_BE_U_4(tp),
637*05b00f60SXin Li 			    GET_BE_U_4(tp + 4));
638*05b00f60SXin Li 			if (optlen > 25) {
639*05b00f60SXin Li 				/* there are sub-options */
640*05b00f60SXin Li 				dhcp6opt_print(ndo, tp + 25, tp + optlen);
641*05b00f60SXin Li 			}
642*05b00f60SXin Li 			ND_PRINT(")");
643*05b00f60SXin Li 			break;
644*05b00f60SXin Li 		case DH6OPT_LIFETIME:
645*05b00f60SXin Li 		case DH6OPT_CLT_TIME:
646*05b00f60SXin Li 			if (optlen != 4) {
647*05b00f60SXin Li 				ND_PRINT(" ?)");
648*05b00f60SXin Li 				break;
649*05b00f60SXin Li 			}
650*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
651*05b00f60SXin Li 			ND_PRINT(" %u)", GET_BE_U_4(tp));
652*05b00f60SXin Li 			break;
653*05b00f60SXin Li 		case DH6OPT_REMOTE_ID:
654*05b00f60SXin Li 			if (optlen < 4) {
655*05b00f60SXin Li 				ND_PRINT(" ?)");
656*05b00f60SXin Li 				break;
657*05b00f60SXin Li 			}
658*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
659*05b00f60SXin Li 			ND_PRINT(" %u ", GET_BE_U_4(tp));
660*05b00f60SXin Li 			/*
661*05b00f60SXin Li 			 * Print hex dump first 10 characters.
662*05b00f60SXin Li 			 */
663*05b00f60SXin Li 			for (i = 4; i < optlen && i < 14; i++)
664*05b00f60SXin Li 				ND_PRINT("%02x", GET_U_1(tp + i));
665*05b00f60SXin Li 			ND_PRINT("...)");
666*05b00f60SXin Li 			break;
667*05b00f60SXin Li 		case DH6OPT_LQ_QUERY:
668*05b00f60SXin Li 			if (optlen < 17) {
669*05b00f60SXin Li 				ND_PRINT(" ?)");
670*05b00f60SXin Li 				break;
671*05b00f60SXin Li 			}
672*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
673*05b00f60SXin Li 			dh6_lq_query_type = GET_U_1(tp);
674*05b00f60SXin Li 			switch (dh6_lq_query_type) {
675*05b00f60SXin Li 			case 1:
676*05b00f60SXin Li 				ND_PRINT(" by-address");
677*05b00f60SXin Li 				break;
678*05b00f60SXin Li 			case 2:
679*05b00f60SXin Li 				ND_PRINT(" by-clientID");
680*05b00f60SXin Li 				break;
681*05b00f60SXin Li 			default:
682*05b00f60SXin Li 				ND_PRINT(" type_%u", dh6_lq_query_type);
683*05b00f60SXin Li 				break;
684*05b00f60SXin Li 			}
685*05b00f60SXin Li 			ND_PRINT(" %s", GET_IP6ADDR_STRING(tp + 1));
686*05b00f60SXin Li 			if (optlen > 17) {
687*05b00f60SXin Li 				/* there are query-options */
688*05b00f60SXin Li 				dhcp6opt_print(ndo, tp + 17, tp + optlen);
689*05b00f60SXin Li 			}
690*05b00f60SXin Li 			ND_PRINT(")");
691*05b00f60SXin Li 			break;
692*05b00f60SXin Li 		case DH6OPT_CLIENT_DATA:
693*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
694*05b00f60SXin Li 			if (optlen > 0) {
695*05b00f60SXin Li 				/* there are encapsulated options */
696*05b00f60SXin Li 				dhcp6opt_print(ndo, tp, tp + optlen);
697*05b00f60SXin Li 			}
698*05b00f60SXin Li 			ND_PRINT(")");
699*05b00f60SXin Li 			break;
700*05b00f60SXin Li 		case DH6OPT_LQ_RELAY_DATA:
701*05b00f60SXin Li 			if (optlen < 16) {
702*05b00f60SXin Li 				ND_PRINT(" ?)");
703*05b00f60SXin Li 				break;
704*05b00f60SXin Li 			}
705*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
706*05b00f60SXin Li 			ND_PRINT(" %s ", GET_IP6ADDR_STRING(tp));
707*05b00f60SXin Li 			/*
708*05b00f60SXin Li 			 * Print hex dump first 10 characters.
709*05b00f60SXin Li 			 */
710*05b00f60SXin Li 			for (i = 16; i < optlen && i < 26; i++)
711*05b00f60SXin Li 				ND_PRINT("%02x", GET_U_1(tp + i));
712*05b00f60SXin Li 			ND_PRINT("...)");
713*05b00f60SXin Li 			break;
714*05b00f60SXin Li 		case DH6OPT_NTP_SERVER:
715*05b00f60SXin Li 			if (optlen < 4) {
716*05b00f60SXin Li 				ND_PRINT(" ?)");
717*05b00f60SXin Li 				break;
718*05b00f60SXin Li 			}
719*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
720*05b00f60SXin Li 			while (tp < cp + sizeof(*dh6o) + optlen - 4) {
721*05b00f60SXin Li 				subopt_code = GET_BE_U_2(tp);
722*05b00f60SXin Li 				tp += 2;
723*05b00f60SXin Li 				subopt_len = GET_BE_U_2(tp);
724*05b00f60SXin Li 				tp += 2;
725*05b00f60SXin Li 				if (tp + subopt_len > cp + sizeof(*dh6o) + optlen)
726*05b00f60SXin Li 					goto trunc;
727*05b00f60SXin Li 				ND_PRINT(" subopt:%u", subopt_code);
728*05b00f60SXin Li 				switch (subopt_code) {
729*05b00f60SXin Li 				case DH6OPT_NTP_SUBOPTION_SRV_ADDR:
730*05b00f60SXin Li 				case DH6OPT_NTP_SUBOPTION_MC_ADDR:
731*05b00f60SXin Li 					if (subopt_len != 16) {
732*05b00f60SXin Li 						ND_PRINT(" ?");
733*05b00f60SXin Li 						break;
734*05b00f60SXin Li 					}
735*05b00f60SXin Li 					ND_PRINT(" %s", GET_IP6ADDR_STRING(tp));
736*05b00f60SXin Li 					break;
737*05b00f60SXin Li 				case DH6OPT_NTP_SUBOPTION_SRV_FQDN:
738*05b00f60SXin Li 					ND_PRINT(" ");
739*05b00f60SXin Li 					if (fqdn_print(ndo, tp, tp + subopt_len) == NULL)
740*05b00f60SXin Li 						goto trunc;
741*05b00f60SXin Li 					break;
742*05b00f60SXin Li 				default:
743*05b00f60SXin Li 					ND_PRINT(" ?");
744*05b00f60SXin Li 					break;
745*05b00f60SXin Li 				}
746*05b00f60SXin Li 				tp += subopt_len;
747*05b00f60SXin Li 			}
748*05b00f60SXin Li 			ND_PRINT(")");
749*05b00f60SXin Li 			break;
750*05b00f60SXin Li 		case DH6OPT_AFTR_NAME:
751*05b00f60SXin Li 			if (optlen < 3) {
752*05b00f60SXin Li 				ND_PRINT(" ?)");
753*05b00f60SXin Li 				break;
754*05b00f60SXin Li 			}
755*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
756*05b00f60SXin Li 			remain_len = optlen;
757*05b00f60SXin Li 			ND_PRINT(" ");
758*05b00f60SXin Li 			/* Encoding is described in section 3.1 of RFC 1035 */
759*05b00f60SXin Li 			while (remain_len && GET_U_1(tp)) {
760*05b00f60SXin Li 				label_len = GET_U_1(tp);
761*05b00f60SXin Li 				tp++;
762*05b00f60SXin Li 				if (label_len < remain_len - 1) {
763*05b00f60SXin Li 					nd_printjnp(ndo, tp, label_len);
764*05b00f60SXin Li 					tp += label_len;
765*05b00f60SXin Li 					remain_len -= (label_len + 1);
766*05b00f60SXin Li 					if(GET_U_1(tp)) ND_PRINT(".");
767*05b00f60SXin Li 				} else {
768*05b00f60SXin Li 					ND_PRINT(" ?");
769*05b00f60SXin Li 					break;
770*05b00f60SXin Li 				}
771*05b00f60SXin Li 			}
772*05b00f60SXin Li 			ND_PRINT(")");
773*05b00f60SXin Li 			break;
774*05b00f60SXin Li 		case DH6OPT_NEW_POSIX_TIMEZONE: /* all three of these options */
775*05b00f60SXin Li 		case DH6OPT_NEW_TZDB_TIMEZONE:	/* are encoded similarly */
776*05b00f60SXin Li 		case DH6OPT_MUDURL:		/* although GMT might not work */
777*05b00f60SXin Li 		        if (optlen < 5) {
778*05b00f60SXin Li 				ND_PRINT(" ?)");
779*05b00f60SXin Li 				break;
780*05b00f60SXin Li 			}
781*05b00f60SXin Li 			tp = (const u_char *)(dh6o + 1);
782*05b00f60SXin Li 			ND_PRINT(" ");
783*05b00f60SXin Li 			nd_printjnp(ndo, tp, optlen);
784*05b00f60SXin Li 			ND_PRINT(")");
785*05b00f60SXin Li 			break;
786*05b00f60SXin Li 
787*05b00f60SXin Li 		default:
788*05b00f60SXin Li 			ND_PRINT(")");
789*05b00f60SXin Li 			break;
790*05b00f60SXin Li 		}
791*05b00f60SXin Li 
792*05b00f60SXin Li 		cp += sizeof(*dh6o) + optlen;
793*05b00f60SXin Li 	}
794*05b00f60SXin Li 	return;
795*05b00f60SXin Li 
796*05b00f60SXin Li trunc:
797*05b00f60SXin Li 	nd_print_trunc(ndo);
798*05b00f60SXin Li }
799*05b00f60SXin Li 
800*05b00f60SXin Li /*
801*05b00f60SXin Li  * Print dhcp6 packets
802*05b00f60SXin Li  */
803*05b00f60SXin Li void
dhcp6_print(netdissect_options * ndo,const u_char * cp,u_int length)804*05b00f60SXin Li dhcp6_print(netdissect_options *ndo,
805*05b00f60SXin Li             const u_char *cp, u_int length)
806*05b00f60SXin Li {
807*05b00f60SXin Li 	const struct dhcp6 *dh6;
808*05b00f60SXin Li 	const struct dhcp6_relay *dh6relay;
809*05b00f60SXin Li 	uint8_t msgtype;
810*05b00f60SXin Li 	const u_char *ep;
811*05b00f60SXin Li 	const u_char *extp;
812*05b00f60SXin Li 	const char *name;
813*05b00f60SXin Li 
814*05b00f60SXin Li 	ndo->ndo_protocol = "dhcp6";
815*05b00f60SXin Li 	ND_PRINT("dhcp6");
816*05b00f60SXin Li 
817*05b00f60SXin Li 	ep = ndo->ndo_snapend;
818*05b00f60SXin Li 	if (cp + length < ep)
819*05b00f60SXin Li 		ep = cp + length;
820*05b00f60SXin Li 
821*05b00f60SXin Li 	dh6 = (const struct dhcp6 *)cp;
822*05b00f60SXin Li 	dh6relay = (const struct dhcp6_relay *)cp;
823*05b00f60SXin Li 	ND_TCHECK_4(dh6->dh6_msgtypexid.xid);
824*05b00f60SXin Li 	msgtype = GET_U_1(dh6->dh6_msgtypexid.msgtype);
825*05b00f60SXin Li 	name = tok2str(dh6_msgtype_str, "msgtype-%u", msgtype);
826*05b00f60SXin Li 
827*05b00f60SXin Li 	if (!ndo->ndo_vflag) {
828*05b00f60SXin Li 		ND_PRINT(" %s", name);
829*05b00f60SXin Li 		return;
830*05b00f60SXin Li 	}
831*05b00f60SXin Li 
832*05b00f60SXin Li 	/* XXX relay agent messages have to be handled differently */
833*05b00f60SXin Li 
834*05b00f60SXin Li 	ND_PRINT(" %s (", name);	/*)*/
835*05b00f60SXin Li 	if (msgtype != DH6_RELAY_FORW && msgtype != DH6_RELAY_REPLY) {
836*05b00f60SXin Li 		ND_PRINT("xid=%x",
837*05b00f60SXin Li 			 GET_BE_U_4(dh6->dh6_msgtypexid.xid) & DH6_XIDMASK);
838*05b00f60SXin Li 		extp = (const u_char *)(dh6 + 1);
839*05b00f60SXin Li 		dhcp6opt_print(ndo, extp, ep);
840*05b00f60SXin Li 	} else {		/* relay messages */
841*05b00f60SXin Li 		ND_PRINT("linkaddr=%s", GET_IP6ADDR_STRING(dh6relay->dh6relay_linkaddr));
842*05b00f60SXin Li 
843*05b00f60SXin Li 		ND_PRINT(" peeraddr=%s", GET_IP6ADDR_STRING(dh6relay->dh6relay_peeraddr));
844*05b00f60SXin Li 
845*05b00f60SXin Li 		dhcp6opt_print(ndo, (const u_char *)(dh6relay + 1), ep);
846*05b00f60SXin Li 	}
847*05b00f60SXin Li 	/*(*/
848*05b00f60SXin Li 	ND_PRINT(")");
849*05b00f60SXin Li 	return;
850*05b00f60SXin Li 
851*05b00f60SXin Li trunc:
852*05b00f60SXin Li 	nd_print_trunc(ndo);
853*05b00f60SXin Li }
854