print-lspping.c revision 147899
1/*
2 * Redistribution and use in source and binary forms, with or without
3 * modification, are permitted provided that: (1) source code
4 * distributions retain the above copyright notice and this paragraph
5 * in its entirety, and (2) distributions including binary code include
6 * the above copyright notice and this paragraph in its entirety in
7 * the documentation or other materials provided with the distribution.
8 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND
9 * WITHOUT ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, WITHOUT
10 * LIMITATION, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
11 * FOR A PARTICULAR PURPOSE.
12 *
13 * Original code by Hannes Gredler (hannes@juniper.net)
14 */
15
16#ifndef lint
17static const char rcsid[] _U_ =
18    "@(#) $Header: /tcpdump/master/tcpdump/print-lspping.c,v 1.12.2.3 2005/05/03 08:12:31 hannes Exp $";
19#endif
20
21#ifdef HAVE_CONFIG_H
22#include "config.h"
23#endif
24
25#include <tcpdump-stdinc.h>
26
27#include <stdio.h>
28#include <stdlib.h>
29#include <string.h>
30
31#include "interface.h"
32#include "extract.h"
33#include "addrtoname.h"
34
35#include "bgp.h"
36#include "l2vpn.h"
37
38/*
39 * LSPPING common header
40 *
41 *  0                   1                   2                   3
42 *  0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
43 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
44 * |         Version Number        |         Must Be Zero          |
45 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
46 * |  Message Type |   Reply mode  |  Return Code  | Return Subcode|
47 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
48 * |                        Sender's Handle                        |
49 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
50 * |                        Sequence Number                        |
51 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
52 * |                    TimeStamp Sent (seconds)                   |
53 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
54 * |                  TimeStamp Sent (microseconds)                |
55 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
56 * |                  TimeStamp Received (seconds)                 |
57 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
58 * |                TimeStamp Received (microseconds)              |
59 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
60 * |                            TLVs ...                           |
61 * .                                                               .
62 * .                                                               .
63 * .                                                               .
64 */
65
66struct lspping_common_header {
67    u_int8_t version[2];
68    u_int8_t reserved[2];
69    u_int8_t msg_type;
70    u_int8_t reply_mode;
71    u_int8_t return_code;
72    u_int8_t return_subcode;
73    u_int8_t sender_handle[4];
74    u_int8_t seq_number[4];
75    u_int8_t ts_sent_sec[4];
76    u_int8_t ts_sent_usec[4];
77    u_int8_t ts_rcvd_sec[4];
78    u_int8_t ts_rcvd_usec[4];
79};
80
81#define LSPPING_VERSION            1
82#define FALSE 0
83#define TRUE  1
84
85static const struct tok lspping_msg_type_values[] = {
86    { 1, "MPLS Echo Request"},
87    { 2, "MPLS Echo Reply"},
88    { 0, NULL}
89};
90
91static const struct tok lspping_reply_mode_values[] = {
92    { 1, "Do not reply"},
93    { 2, "Reply via an IPv4/IPv6 UDP packet"},
94    { 3, "Reply via an IPv4/IPv6 UDP packet with Router Alert"},
95    { 4, "Reply via application level control channel"},
96    { 0, NULL}
97};
98
99static const struct tok lspping_return_code_values[] = {
100    {  0, "No return code or return code contained in the Error Code TLV"},
101    {  1, "Malformed echo request received"},
102    {  2, "One or more of the TLVs was not understood"},
103    {  3, "Replying router is an egress for the FEC at stack depth"},
104    {  4, "Replying router has no mapping for the FEC at stack depth"},
105    {  5, "Reserved"},
106    {  6, "Reserved"},
107    {  7, "Reserved"},
108    {  8, "Label switched at stack-depth"},
109    {  9, "Label switched but no MPLS forwarding at stack-depth"},
110    { 10, "Mapping for this FEC is not the given label at stack depth"},
111    { 11, "No label entry at stack-depth"},
112    { 12, "Protocol not associated with interface at FEC stack depth"},
113};
114
115
116/*
117 * LSPPING TLV header
118 *  0                   1                   2                   3
119 *  0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
120 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
121 * |             Type              |            Length             |
122 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
123 * |                             Value                             |
124 * .                                                               .
125 * .                                                               .
126 * .                                                               .
127 * |                                                               |
128 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
129 */
130
131struct lspping_tlv_header {
132    u_int8_t type[2];
133    u_int8_t length[2];
134};
135
136#define	LSPPING_TLV_TARGET_FEC_STACK      1
137#define	LSPPING_TLV_DOWNSTREAM_MAPPING    2
138#define	LSPPING_TLV_PAD                   3
139#define	LSPPING_TLV_ERROR_CODE            4
140#define	LSPPING_TLV_VENDOR_PRIVATE        0xfc00
141
142static const struct tok lspping_tlv_values[] = {
143    { LSPPING_TLV_TARGET_FEC_STACK, "Target FEC Stack" },
144    { LSPPING_TLV_DOWNSTREAM_MAPPING, "Downstream Mapping" },
145    { LSPPING_TLV_PAD, "Pad" },
146    { LSPPING_TLV_ERROR_CODE, "Error Code" },
147    { LSPPING_TLV_VENDOR_PRIVATE, "Vendor Enterprise Code" },
148    { 0, NULL}
149};
150
151#define	LSPPING_TLV_TARGETFEC_SUBTLV_LDP_IPV4      1
152#define	LSPPING_TLV_TARGETFEC_SUBTLV_LDP_IPV6      2
153#define	LSPPING_TLV_TARGETFEC_SUBTLV_RSVP_IPV4     3
154#define	LSPPING_TLV_TARGETFEC_SUBTLV_RSVP_IPV6     4
155#define	LSPPING_TLV_TARGETFEC_SUBTLV_L3VPN_IPV4    6
156#define	LSPPING_TLV_TARGETFEC_SUBTLV_L3VPN_IPV6    7
157#define	LSPPING_TLV_TARGETFEC_SUBTLV_L2VPN_ENDPT   8
158#define	LSPPING_TLV_TARGETFEC_SUBTLV_L2VPN_VCID_OLD 9
159#define	LSPPING_TLV_TARGETFEC_SUBTLV_L2VPN_VCID   10
160#define	LSPPING_TLV_TARGETFEC_SUBTLV_BGP_IPV4     11
161#define	LSPPING_TLV_TARGETFEC_SUBTLV_BGP_IPV6     12
162
163static const struct tok lspping_tlvtargetfec_subtlv_values[] = {
164    { LSPPING_TLV_TARGETFEC_SUBTLV_LDP_IPV4, "LDP IPv4 prefix"},
165    { LSPPING_TLV_TARGETFEC_SUBTLV_LDP_IPV6, "LDP IPv6 prefix"},
166    { LSPPING_TLV_TARGETFEC_SUBTLV_RSVP_IPV4, "RSVP IPv4 Session Query"},
167    { LSPPING_TLV_TARGETFEC_SUBTLV_RSVP_IPV6, "RSVP IPv6 Session Query"},
168    { 5, "Reserved"},
169    { LSPPING_TLV_TARGETFEC_SUBTLV_L3VPN_IPV4, "VPN IPv4 prefix"},
170    { LSPPING_TLV_TARGETFEC_SUBTLV_L3VPN_IPV6, "VPN IPv6 prefix"},
171    { LSPPING_TLV_TARGETFEC_SUBTLV_L2VPN_ENDPT, "L2 VPN endpoint"},
172    { LSPPING_TLV_TARGETFEC_SUBTLV_L2VPN_VCID_OLD, "L2 circuit ID (old)"},
173    { LSPPING_TLV_TARGETFEC_SUBTLV_L2VPN_VCID, "L2 circuit ID"},
174    { LSPPING_TLV_TARGETFEC_SUBTLV_BGP_IPV4, "BGP labeled IPv4 prefix"},
175    { LSPPING_TLV_TARGETFEC_SUBTLV_BGP_IPV6, "BGP labeled IPv6 prefix"},
176    { 0, NULL}
177};
178
179/*
180 *  0                   1                   2                   3
181 *  0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
182 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
183 * |                          IPv4 prefix                          |
184 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
185 * | Prefix Length |         Must Be Zero                          |
186 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
187 */
188struct lspping_tlv_targetfec_subtlv_ldp_ipv4_t {
189    u_int8_t prefix [4];
190    u_int8_t prefix_len;
191};
192
193/*
194 *  0                   1                   2                   3
195 *  0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
196 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
197 * |                          IPv6 prefix                          |
198 * |                          (16 octets)                          |
199 * |                                                               |
200 * |                                                               |
201 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
202 * | Prefix Length |         Must Be Zero                          |
203 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
204 */
205struct lspping_tlv_targetfec_subtlv_ldp_ipv6_t {
206    u_int8_t prefix [16];
207    u_int8_t prefix_len;
208};
209
210/*
211 * 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
212 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
213 * |                    Sender identifier                          |
214 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
215 * |                         IPv4 prefix                           |
216 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
217 * | Prefix Length |                 Must Be Zero                  |
218 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
219 */
220struct lspping_tlv_targetfec_subtlv_bgp_ipv4_t {
221    u_int8_t sender_id [4];
222    u_int8_t prefix [4];
223    u_int8_t prefix_len;
224};
225
226/*
227 * 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
228 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
229 * |                    Sender identifier                          |
230 * |                          (16 octets)                          |
231 * |                                                               |
232 * |                                                               |
233 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
234 * |                          IPv6 prefix                          |
235 * |                          (16 octets)                          |
236 * |                                                               |
237 * |                                                               |
238 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
239 * | Prefix Length |                 Must Be Zero                  |
240 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
241 */
242struct lspping_tlv_targetfec_subtlv_bgp_ipv6_t {
243    u_int8_t sender_id [16];
244    u_int8_t prefix [16];
245    u_int8_t prefix_len;
246};
247
248/*
249 *  0                   1                   2                   3
250 *  0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
251 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
252 * |                 IPv4 tunnel end point address                 |
253 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
254 * |          Must Be Zero         |     Tunnel ID                 |
255 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
256 * |                       Extended Tunnel ID                      |
257 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
258 * |                   IPv4 tunnel sender address                  |
259 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
260 * |          Must Be Zero         |            LSP ID             |
261 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
262 */
263struct lspping_tlv_targetfec_subtlv_rsvp_ipv4_t {
264    u_int8_t tunnel_endpoint [4];
265    u_int8_t res[2];
266    u_int8_t tunnel_id[2];
267    u_int8_t extended_tunnel_id[4];
268    u_int8_t tunnel_sender [4];
269    u_int8_t res2[2];
270    u_int8_t lsp_id [2];
271};
272
273/*
274 *  0                   1                   2                   3
275 *  0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
276 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
277 * |                 IPv6 tunnel end point address                 |
278 * |                                                               |
279 * |                                                               |
280 * |                                                               |
281 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
282 * |          Must Be Zero         |          Tunnel ID            |
283 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
284 * |                       Extended Tunnel ID                      |
285 * |                                                               |
286 * |                                                               |
287 * |                                                               |
288 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
289 * |                   IPv6 tunnel sender address                  |
290 * |                                                               |
291 * |                                                               |
292 * |                                                               |
293 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
294 * |          Must Be Zero         |            LSP ID             |
295 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
296 */
297struct lspping_tlv_targetfec_subtlv_rsvp_ipv6_t {
298    u_int8_t tunnel_endpoint [16];
299    u_int8_t res[2];
300    u_int8_t tunnel_id[2];
301    u_int8_t extended_tunnel_id[16];
302    u_int8_t tunnel_sender [16];
303    u_int8_t res2[2];
304    u_int8_t lsp_id [2];
305};
306
307/*
308 *  0                   1                   2                   3
309 *  0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
310 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
311 * |                      Route Distinguisher                      |
312 * |                          (8 octets)                           |
313 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
314 * |                         IPv4 prefix                           |
315 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
316 * | Prefix Length |                 Must Be Zero                  |
317 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
318 */
319struct lspping_tlv_targetfec_subtlv_l3vpn_ipv4_t {
320    u_int8_t rd [8];
321    u_int8_t prefix [4];
322    u_int8_t prefix_len;
323};
324
325/*
326 *  0                   1                   2                   3
327 *  0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
328 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
329 * |                      Route Distinguisher                      |
330 * |                          (8 octets)                           |
331 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
332 * |                          IPv6 prefix                          |
333 * |                          (16 octets)                          |
334 * |                                                               |
335 * |                                                               |
336 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
337 * | Prefix Length |                 Must Be Zero                  |
338 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
339 */
340struct lspping_tlv_targetfec_subtlv_l3vpn_ipv6_t {
341    u_int8_t rd [8];
342    u_int8_t prefix [16];
343    u_int8_t prefix_len;
344};
345
346/*
347 *  0                   1                   2                   3
348 *  0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
349 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
350 * |                      Route Distinguisher                      |
351 * |                          (8 octets)                           |
352 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
353 * |         Sender's CE ID        |       Receiver's CE ID        |
354 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
355 * |      Encapsulation Type       |         Must Be Zero          |
356 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
357 *  0                   1                   2                   3
358 */
359struct lspping_tlv_targetfec_subtlv_l2vpn_endpt_t {
360    u_int8_t rd [8];
361    u_int8_t sender_ce_id [2];
362    u_int8_t receiver_ce_id [2];
363    u_int8_t encapsulation[2];
364};
365
366/*
367 *  0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
368 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
369 * |                      Remote PE Address                        |
370 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
371 * |                             VC ID                             |
372 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
373 * |      Encapsulation Type       |         Must Be Zero          |
374 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
375 */
376struct lspping_tlv_targetfec_subtlv_l2vpn_vcid_old_t {
377    u_int8_t remote_pe_address [4];
378    u_int8_t vc_id [4];
379    u_int8_t encapsulation[2];
380};
381
382/*
383 *  0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
384 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
385 * |                     Sender's PE Address                       |
386 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
387 * |                      Remote PE Address                        |
388 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
389 * |                             VC ID                             |
390 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
391 * |      Encapsulation Type       |         Must Be Zero          |
392 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
393 */
394struct lspping_tlv_targetfec_subtlv_l2vpn_vcid_t {
395    u_int8_t sender_pe_address [4];
396    u_int8_t remote_pe_address [4];
397    u_int8_t vc_id [4];
398    u_int8_t encapsulation[2];
399};
400
401/*
402 *  0                   1                   2                   3
403 *  0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
404 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
405 * |               MTU             | Address Type  |  Resvd (SBZ)  |
406 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
407 * |             Downstream IP Address (4 or 16 octets)            |
408 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
409 * |         Downstream Interface Address (4 or 16 octets)         |
410 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
411 * | Hash Key Type | Depth Limit   |        Multipath Length       |
412 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
413 * .                                                               .
414 * .                     (Multipath Information)                   .
415 * .                                                               .
416 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
417 * |               Downstream Label                |    Protocol   |
418 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
419 * .                                                               .
420 * .                                                               .
421 * .                                                               .
422 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
423 * |               Downstream Label                |    Protocol   |
424 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
425 */
426struct lspping_tlv_downstream_map_ipv4_t {
427    u_int8_t mtu [2];
428    u_int8_t address_type;
429    u_int8_t res;
430    u_int8_t downstream_ip[4];
431    u_int8_t downstream_interface[4];
432};
433
434struct lspping_tlv_downstream_map_ipv6_t {
435    u_int8_t mtu [2];
436    u_int8_t address_type;
437    u_int8_t res;
438    u_int8_t downstream_ip[16];
439    u_int8_t downstream_interface[16];
440};
441
442struct lspping_tlv_downstream_map_info_t {
443    u_int8_t hash_key_type;
444    u_int8_t depth_limit;
445    u_int8_t multipath_length [2];
446};
447
448#define LSPPING_AFI_IPV4 1
449#define LSPPING_AFI_UNMB 2
450#define LSPPING_AFI_IPV6 3
451
452static const struct tok lspping_tlv_downstream_addr_values[] = {
453    { LSPPING_AFI_IPV4, "IPv4"},
454    { LSPPING_AFI_IPV6, "IPv6"},
455    { LSPPING_AFI_UNMB, "Unnumbered"},
456    { 0, NULL}
457};
458
459void
460lspping_print(register const u_char *pptr, register u_int len) {
461
462    const struct lspping_common_header *lspping_com_header;
463    const struct lspping_tlv_header *lspping_tlv_header;
464    const struct lspping_tlv_header *lspping_subtlv_header;
465    const u_char *tptr,*tlv_tptr,*subtlv_tptr;
466    int tlen,lspping_tlv_len,lspping_tlv_type,tlv_tlen;
467    int tlv_hexdump,subtlv_hexdump;
468    int lspping_subtlv_len,lspping_subtlv_type;
469    struct timeval timestamp;
470
471    union {
472        const struct lspping_tlv_downstream_map_ipv4_t *lspping_tlv_downstream_map_ipv4;
473        const struct lspping_tlv_downstream_map_ipv6_t *lspping_tlv_downstream_map_ipv6;
474        const struct lspping_tlv_downstream_map_info_t  *lspping_tlv_downstream_map_info;
475    } tlv_ptr;
476
477    union {
478        const struct lspping_tlv_targetfec_subtlv_ldp_ipv4_t *lspping_tlv_targetfec_subtlv_ldp_ipv4;
479        const struct lspping_tlv_targetfec_subtlv_ldp_ipv6_t *lspping_tlv_targetfec_subtlv_ldp_ipv6;
480        const struct lspping_tlv_targetfec_subtlv_rsvp_ipv4_t *lspping_tlv_targetfec_subtlv_rsvp_ipv4;
481        const struct lspping_tlv_targetfec_subtlv_rsvp_ipv6_t *lspping_tlv_targetfec_subtlv_rsvp_ipv6;
482        const struct lspping_tlv_targetfec_subtlv_l3vpn_ipv4_t *lspping_tlv_targetfec_subtlv_l3vpn_ipv4;
483        const struct lspping_tlv_targetfec_subtlv_l3vpn_ipv6_t *lspping_tlv_targetfec_subtlv_l3vpn_ipv6;
484        const struct lspping_tlv_targetfec_subtlv_l2vpn_endpt_t *lspping_tlv_targetfec_subtlv_l2vpn_endpt;
485        const struct lspping_tlv_targetfec_subtlv_l2vpn_vcid_old_t *lspping_tlv_targetfec_subtlv_l2vpn_vcid_old;
486        const struct lspping_tlv_targetfec_subtlv_l2vpn_vcid_t *lspping_tlv_targetfec_subtlv_l2vpn_vcid;
487        const struct lspping_tlv_targetfec_subtlv_bgp_ipv4_t *lspping_tlv_targetfec_subtlv_bgp_ipv4;
488        const struct lspping_tlv_targetfec_subtlv_bgp_ipv6_t *lspping_tlv_targetfec_subtlv_bgp_ipv6;
489    } subtlv_ptr;
490
491    tptr=pptr;
492    lspping_com_header = (const struct lspping_common_header *)pptr;
493    TCHECK(*lspping_com_header);
494
495    /*
496     * Sanity checking of the header.
497     */
498    if (EXTRACT_16BITS(&lspping_com_header->version[0]) != LSPPING_VERSION) {
499	printf("LSP-PING version %u packet not supported",
500               EXTRACT_16BITS(&lspping_com_header->version[0]));
501	return;
502    }
503
504    /* in non-verbose mode just lets print the basic Message Type*/
505    if (vflag < 1) {
506        printf("LSP-PINGv%u, %s, seq %u, length: %u",
507               EXTRACT_16BITS(&lspping_com_header->version[0]),
508               tok2str(lspping_msg_type_values, "unknown (%u)",lspping_com_header->msg_type),
509               EXTRACT_32BITS(lspping_com_header->seq_number),
510               len);
511        return;
512    }
513
514    /* ok they seem to want to know everything - lets fully decode it */
515
516    tlen=len;
517
518    printf("\n\tLSP-PINGv%u, msg-type: %s (%u), length: %u\n\t  reply-mode: %s (%u)",
519           EXTRACT_16BITS(&lspping_com_header->version[0]),
520           tok2str(lspping_msg_type_values, "unknown",lspping_com_header->msg_type),
521           lspping_com_header->msg_type,
522           len,
523           tok2str(lspping_reply_mode_values, "unknown",lspping_com_header->reply_mode),
524           lspping_com_header->reply_mode);
525
526    /*
527     *  the following return codes require that the subcode is attached
528     *  at the end of the translated token output
529     */
530    if (lspping_com_header->return_code == 3 ||
531        lspping_com_header->return_code == 4 ||
532        lspping_com_header->return_code == 8 ||
533        lspping_com_header->return_code == 10 ||
534        lspping_com_header->return_code == 11 ||
535        lspping_com_header->return_code == 12 )
536        printf("\n\t  Return Code: %s %u (%u)\n\t  Return Subcode: (%u)",
537               tok2str(lspping_return_code_values, "unknown",lspping_com_header->return_code),
538               lspping_com_header->return_subcode,
539               lspping_com_header->return_code,
540               lspping_com_header->return_subcode);
541    else
542        printf("\n\t  Return Code: %s (%u)\n\t  Return Subcode: (%u)",
543               tok2str(lspping_return_code_values, "unknown",lspping_com_header->return_code),
544               lspping_com_header->return_code,
545               lspping_com_header->return_subcode);
546
547    printf("\n\t  Sender Handle: 0x%08x, Sequence: %u",
548           EXTRACT_32BITS(lspping_com_header->sender_handle),
549           EXTRACT_32BITS(lspping_com_header->seq_number));
550
551    timestamp.tv_sec=EXTRACT_32BITS(lspping_com_header->ts_sent_sec);
552    timestamp.tv_usec=EXTRACT_32BITS(lspping_com_header->ts_sent_usec);
553    printf("\n\t  Sender Timestamp: ");
554    ts_print(&timestamp);
555
556    timestamp.tv_sec=EXTRACT_32BITS(lspping_com_header->ts_rcvd_sec);
557    timestamp.tv_usec=EXTRACT_32BITS(lspping_com_header->ts_rcvd_usec);
558    printf("Receiver Timestamp: ");
559    if ((timestamp.tv_sec != 0) && (timestamp.tv_usec != 0))
560        ts_print(&timestamp);
561    else
562        printf("no timestamp");
563
564    tptr+=sizeof(const struct lspping_common_header);
565    tlen-=sizeof(const struct lspping_common_header);
566
567    while(tlen>(int)sizeof(struct lspping_tlv_header)) {
568        /* did we capture enough for fully decoding the tlv header ? */
569        if (!TTEST2(*tptr, sizeof(struct lspping_tlv_header)))
570            goto trunc;
571
572        lspping_tlv_header = (const struct lspping_tlv_header *)tptr;
573        lspping_tlv_type=EXTRACT_16BITS(lspping_tlv_header->type);
574        lspping_tlv_len=EXTRACT_16BITS(lspping_tlv_header->length);
575
576        if (lspping_tlv_len == 0)
577            return;
578
579        if(lspping_tlv_len % 4 || lspping_tlv_len < 4) { /* aligned to four octet boundary */
580            printf("\n\t  ERROR: TLV %u bogus size %u",lspping_tlv_type,lspping_tlv_len);
581            return;
582        }
583
584        printf("\n\t  %s TLV (%u), length: %u",
585               tok2str(lspping_tlv_values,
586                       "Unknown",
587                       lspping_tlv_type),
588               lspping_tlv_type,
589               lspping_tlv_len);
590
591        tlv_tptr=tptr+sizeof(struct lspping_tlv_header);
592        tlv_tlen=lspping_tlv_len; /* header not included -> no adjustment */
593
594        /* did we capture enough for fully decoding the tlv ? */
595        if (!TTEST2(*tptr, lspping_tlv_len))
596            goto trunc;
597        tlv_hexdump=FALSE;
598
599        switch(lspping_tlv_type) {
600        case LSPPING_TLV_TARGET_FEC_STACK:
601            while(tlv_tlen>(int)sizeof(struct lspping_tlv_header)) {
602
603                /* did we capture enough for fully decoding the subtlv header ? */
604                if (!TTEST2(*tptr, sizeof(struct lspping_tlv_header)))
605                    goto trunc;
606                subtlv_hexdump=FALSE;
607
608                lspping_subtlv_header = (const struct lspping_tlv_header *)tlv_tptr;
609                lspping_subtlv_type=EXTRACT_16BITS(lspping_subtlv_header->type);
610                lspping_subtlv_len=EXTRACT_16BITS(lspping_subtlv_header->length);
611                subtlv_tptr=tlv_tptr+sizeof(struct lspping_tlv_header);
612
613                if (lspping_subtlv_len == 0)
614                    break;
615
616                printf("\n\t    %s subTLV (%u), length: %u",
617                       tok2str(lspping_tlvtargetfec_subtlv_values,
618                               "Unknown",
619                               lspping_subtlv_type),
620                       lspping_subtlv_type,
621                       lspping_subtlv_len);
622
623                switch(lspping_subtlv_type) {
624
625                case LSPPING_TLV_TARGETFEC_SUBTLV_LDP_IPV4:
626                    subtlv_ptr.lspping_tlv_targetfec_subtlv_ldp_ipv4 = \
627                        (const struct lspping_tlv_targetfec_subtlv_ldp_ipv4_t *)subtlv_tptr;
628                    printf("\n\t      %s/%u",
629                           ipaddr_string(subtlv_ptr.lspping_tlv_targetfec_subtlv_ldp_ipv4->prefix),
630                           subtlv_ptr.lspping_tlv_targetfec_subtlv_ldp_ipv4->prefix_len);
631                    break;
632
633#ifdef INET6
634                case LSPPING_TLV_TARGETFEC_SUBTLV_LDP_IPV6:
635                    subtlv_ptr.lspping_tlv_targetfec_subtlv_ldp_ipv6 = \
636                        (const struct lspping_tlv_targetfec_subtlv_ldp_ipv6_t *)subtlv_tptr;
637                    printf("\n\t      %s/%u",
638                           ip6addr_string(subtlv_ptr.lspping_tlv_targetfec_subtlv_ldp_ipv6->prefix),
639                           subtlv_ptr.lspping_tlv_targetfec_subtlv_ldp_ipv6->prefix_len);
640                    break;
641#endif
642
643                case LSPPING_TLV_TARGETFEC_SUBTLV_BGP_IPV4:
644                    subtlv_ptr.lspping_tlv_targetfec_subtlv_bgp_ipv4 = \
645                        (const struct lspping_tlv_targetfec_subtlv_bgp_ipv4_t *)subtlv_tptr;
646                    printf("\n\t      %s/%u, sender-id %s",
647                           ipaddr_string(subtlv_ptr.lspping_tlv_targetfec_subtlv_bgp_ipv4->prefix),
648                           subtlv_ptr.lspping_tlv_targetfec_subtlv_bgp_ipv4->prefix_len,
649                           ipaddr_string(subtlv_ptr.lspping_tlv_targetfec_subtlv_bgp_ipv4->sender_id));
650                    break;
651
652#ifdef INET6
653                case LSPPING_TLV_TARGETFEC_SUBTLV_BGP_IPV6:
654                    subtlv_ptr.lspping_tlv_targetfec_subtlv_bgp_ipv6 = \
655                        (const struct lspping_tlv_targetfec_subtlv_bgp_ipv6_t *)subtlv_tptr;
656                    printf("\n\t      %s/%u, sender-id %s",
657                           ip6addr_string(subtlv_ptr.lspping_tlv_targetfec_subtlv_bgp_ipv6->prefix),
658                           subtlv_ptr.lspping_tlv_targetfec_subtlv_bgp_ipv6->prefix_len,
659                           ip6addr_string(subtlv_ptr.lspping_tlv_targetfec_subtlv_bgp_ipv6->sender_id));
660                    break;
661#endif
662
663                case LSPPING_TLV_TARGETFEC_SUBTLV_RSVP_IPV4:
664                    subtlv_ptr.lspping_tlv_targetfec_subtlv_rsvp_ipv4 = \
665                        (const struct lspping_tlv_targetfec_subtlv_rsvp_ipv4_t *)subtlv_tptr;
666                    printf("\n\t      tunnel end-point %s, tunnel sender %s, lsp-id 0x%04x" \
667                           "\n\t      tunnel-id 0x%04x, extended tunnel-id %s",
668                           ipaddr_string(subtlv_ptr.lspping_tlv_targetfec_subtlv_rsvp_ipv4->tunnel_endpoint),
669                           ipaddr_string(subtlv_ptr.lspping_tlv_targetfec_subtlv_rsvp_ipv4->tunnel_sender),
670                           EXTRACT_16BITS(subtlv_ptr.lspping_tlv_targetfec_subtlv_rsvp_ipv4->lsp_id),
671                           EXTRACT_16BITS(subtlv_ptr.lspping_tlv_targetfec_subtlv_rsvp_ipv4->tunnel_id),
672                           ipaddr_string(subtlv_ptr.lspping_tlv_targetfec_subtlv_rsvp_ipv4->extended_tunnel_id));
673                    break;
674
675#ifdef INET6
676                case LSPPING_TLV_TARGETFEC_SUBTLV_RSVP_IPV6:
677                    subtlv_ptr.lspping_tlv_targetfec_subtlv_rsvp_ipv6 = \
678                        (const struct lspping_tlv_targetfec_subtlv_rsvp_ipv6_t *)subtlv_tptr;
679                    printf("\n\t      tunnel end-point %s, tunnel sender %s, lsp-id 0x%04x" \
680                           "\n\t      tunnel-id 0x%04x, extended tunnel-id %s",
681                           ip6addr_string(subtlv_ptr.lspping_tlv_targetfec_subtlv_rsvp_ipv6->tunnel_endpoint),
682                           ip6addr_string(subtlv_ptr.lspping_tlv_targetfec_subtlv_rsvp_ipv6->tunnel_sender),
683                           EXTRACT_16BITS(subtlv_ptr.lspping_tlv_targetfec_subtlv_rsvp_ipv6->lsp_id),
684                           EXTRACT_16BITS(subtlv_ptr.lspping_tlv_targetfec_subtlv_rsvp_ipv6->tunnel_id),
685                           ip6addr_string(subtlv_ptr.lspping_tlv_targetfec_subtlv_rsvp_ipv6->extended_tunnel_id));
686                    break;
687#endif
688
689                case LSPPING_TLV_TARGETFEC_SUBTLV_L3VPN_IPV4:
690                    subtlv_ptr.lspping_tlv_targetfec_subtlv_l3vpn_ipv4 = \
691                        (const struct lspping_tlv_targetfec_subtlv_l3vpn_ipv4_t *)subtlv_tptr;
692                    printf("\n\t      RD: %s, %s/%u",
693                           bgp_vpn_rd_print(subtlv_ptr.lspping_tlv_targetfec_subtlv_l3vpn_ipv4->rd),
694                           ipaddr_string(subtlv_ptr.lspping_tlv_targetfec_subtlv_l3vpn_ipv4->prefix),
695                           subtlv_ptr.lspping_tlv_targetfec_subtlv_l3vpn_ipv4->prefix_len);
696                    break;
697
698#ifdef INET6
699                case LSPPING_TLV_TARGETFEC_SUBTLV_L3VPN_IPV6:
700                    subtlv_ptr.lspping_tlv_targetfec_subtlv_l3vpn_ipv6 = \
701                        (const struct lspping_tlv_targetfec_subtlv_l3vpn_ipv6_t *)subtlv_tptr;
702                    printf("\n\t      RD: %s, %s/%u",
703                           bgp_vpn_rd_print(subtlv_ptr.lspping_tlv_targetfec_subtlv_l3vpn_ipv6->rd),
704                           ip6addr_string(subtlv_ptr.lspping_tlv_targetfec_subtlv_l3vpn_ipv6->prefix),
705                           subtlv_ptr.lspping_tlv_targetfec_subtlv_l3vpn_ipv6->prefix_len);
706                    break;
707#endif
708
709                case LSPPING_TLV_TARGETFEC_SUBTLV_L2VPN_ENDPT:
710                    subtlv_ptr.lspping_tlv_targetfec_subtlv_l2vpn_endpt = \
711                        (const struct lspping_tlv_targetfec_subtlv_l2vpn_endpt_t *)subtlv_tptr;
712                    printf("\n\t      RD: %s, Sender CE-ID: %u, Receiver CE-ID: %u" \
713                           "\n\t      Encapsulation Type: %s (%u)",
714                           bgp_vpn_rd_print(subtlv_ptr.lspping_tlv_targetfec_subtlv_l2vpn_endpt->rd),
715                           EXTRACT_16BITS(subtlv_ptr.lspping_tlv_targetfec_subtlv_l2vpn_endpt->sender_ce_id),
716                           EXTRACT_16BITS(subtlv_ptr.lspping_tlv_targetfec_subtlv_l2vpn_endpt->receiver_ce_id),
717                           tok2str(l2vpn_encaps_values,
718                                   "unknown",
719                                   EXTRACT_16BITS(subtlv_ptr.lspping_tlv_targetfec_subtlv_l2vpn_endpt->encapsulation)),
720                           EXTRACT_16BITS(subtlv_ptr.lspping_tlv_targetfec_subtlv_l2vpn_endpt->encapsulation));
721
722                    break;
723
724                    /* the old L2VPN VCID subTLV does not have support for the sender field */
725                case LSPPING_TLV_TARGETFEC_SUBTLV_L2VPN_VCID_OLD:
726                    subtlv_ptr.lspping_tlv_targetfec_subtlv_l2vpn_vcid_old = \
727                        (const struct lspping_tlv_targetfec_subtlv_l2vpn_vcid_old_t *)subtlv_tptr;
728                    printf("\n\t      Remote PE: %s" \
729                           "\n\t      VC-ID: 0x%08x, Encapsulation Type: %s (%u)",
730                           ipaddr_string(subtlv_ptr.lspping_tlv_targetfec_subtlv_l2vpn_vcid_old->remote_pe_address),
731                           EXTRACT_32BITS(subtlv_ptr.lspping_tlv_targetfec_subtlv_l2vpn_vcid_old->vc_id),
732                           tok2str(l2vpn_encaps_values,
733                                   "unknown",
734                                   EXTRACT_16BITS(subtlv_ptr.lspping_tlv_targetfec_subtlv_l2vpn_vcid_old->encapsulation)),
735                           EXTRACT_16BITS(subtlv_ptr.lspping_tlv_targetfec_subtlv_l2vpn_vcid_old->encapsulation));
736
737                    break;
738
739                case LSPPING_TLV_TARGETFEC_SUBTLV_L2VPN_VCID:
740                    subtlv_ptr.lspping_tlv_targetfec_subtlv_l2vpn_vcid = \
741                        (const struct lspping_tlv_targetfec_subtlv_l2vpn_vcid_t *)subtlv_tptr;
742                    printf("\n\t      Sender PE: %s, Remote PE: %s" \
743                           "\n\t      VC-ID: 0x%08x, Encapsulation Type: %s (%u)",
744                           ipaddr_string(subtlv_ptr.lspping_tlv_targetfec_subtlv_l2vpn_vcid->sender_pe_address),
745                           ipaddr_string(subtlv_ptr.lspping_tlv_targetfec_subtlv_l2vpn_vcid->remote_pe_address),
746                           EXTRACT_32BITS(subtlv_ptr.lspping_tlv_targetfec_subtlv_l2vpn_vcid->vc_id),
747                           tok2str(l2vpn_encaps_values,
748                                   "unknown",
749                                   EXTRACT_16BITS(subtlv_ptr.lspping_tlv_targetfec_subtlv_l2vpn_vcid->encapsulation)),
750                           EXTRACT_16BITS(subtlv_ptr.lspping_tlv_targetfec_subtlv_l2vpn_vcid->encapsulation));
751
752                    break;
753
754                default:
755                    subtlv_hexdump=TRUE; /* unknown subTLV just hexdump it */
756                    break;
757                }
758                /* do we want to see an additionally subtlv hexdump ? */
759                if (vflag > 1 || subtlv_hexdump==TRUE)
760                    print_unknown_data(tlv_tptr+sizeof(struct lspping_tlv_header), \
761                                       "\n\t      ",
762                                       lspping_subtlv_len);
763
764                tlv_tptr+=lspping_subtlv_len;
765                tlv_tlen-=lspping_subtlv_len+sizeof(struct lspping_tlv_header);
766            }
767            break;
768
769        case LSPPING_TLV_DOWNSTREAM_MAPPING:
770            /* that strange thing with the downstream map TLV is that until now
771             * we do not know if its IPv4 or IPv6 , after we found the adress-type
772             * lets recast the tlv_tptr and move on */
773
774            tlv_ptr.lspping_tlv_downstream_map_ipv4= \
775                (const struct lspping_tlv_downstream_map_ipv4_t *)tlv_tptr;
776            tlv_ptr.lspping_tlv_downstream_map_ipv6= \
777                (const struct lspping_tlv_downstream_map_ipv6_t *)tlv_tptr;
778            printf("\n\t    MTU: %u, Address-Type: %s (%u)",
779                   EXTRACT_16BITS(tlv_ptr.lspping_tlv_downstream_map_ipv4->mtu),
780                   tok2str(lspping_tlv_downstream_addr_values,
781                           "unknown",
782                           tlv_ptr.lspping_tlv_downstream_map_ipv4->address_type),
783                   tlv_ptr.lspping_tlv_downstream_map_ipv4->address_type);
784
785            switch(tlv_ptr.lspping_tlv_downstream_map_ipv4->address_type) {
786
787            case LSPPING_AFI_IPV4:
788                printf("\n\t    Downstream IP: %s" \
789                       "\n\t    Downstream Interface IP: %s",
790                       ipaddr_string(tlv_ptr.lspping_tlv_downstream_map_ipv4->downstream_ip),
791                       ipaddr_string(tlv_ptr.lspping_tlv_downstream_map_ipv4->downstream_interface));
792                tlv_tptr+=sizeof(struct lspping_tlv_downstream_map_ipv4_t);
793                tlv_tlen-=sizeof(struct lspping_tlv_downstream_map_ipv4_t);
794                break;
795#ifdef INET6
796             case LSPPING_AFI_IPV6:
797                printf("\n\t    Downstream IP: %s" \
798                       "\n\t    Downstream Interface IP: %s",
799                       ip6addr_string(tlv_ptr.lspping_tlv_downstream_map_ipv6->downstream_ip),
800                       ip6addr_string(tlv_ptr.lspping_tlv_downstream_map_ipv6->downstream_interface));
801                tlv_tptr+=sizeof(struct lspping_tlv_downstream_map_ipv6_t);
802                tlv_tlen-=sizeof(struct lspping_tlv_downstream_map_ipv6_t);
803                break;
804#endif
805            case LSPPING_AFI_UNMB:
806                printf("\n\t    Downstream IP: %s" \
807                       "\n\t    Downstream Interface Index: 0x%08x",
808                       ipaddr_string(tlv_ptr.lspping_tlv_downstream_map_ipv4->downstream_ip),
809                       EXTRACT_32BITS(tlv_ptr.lspping_tlv_downstream_map_ipv4->downstream_interface));
810                tlv_tptr+=sizeof(struct lspping_tlv_downstream_map_ipv4_t);
811                tlv_tlen-=sizeof(struct lspping_tlv_downstream_map_ipv4_t);
812                break;
813
814            default:
815                /* should not happen ! - no error message - tok2str() has barked already */
816                break;
817            }
818
819            tlv_ptr.lspping_tlv_downstream_map_info= \
820                (const struct lspping_tlv_downstream_map_info_t *)tlv_tptr;
821
822            /* FIXME add hash-key type, depth limit, multipath processing */
823
824
825            tlv_tptr+=sizeof(struct lspping_tlv_downstream_map_info_t);
826            tlv_tlen-=sizeof(struct lspping_tlv_downstream_map_info_t);
827
828            /* FIXME print downstream labels */
829
830
831            tlv_hexdump=TRUE; /* dump the TLV until code complete */
832
833            break;
834
835            /*
836             *  FIXME those are the defined TLVs that lack a decoder
837             *  you are welcome to contribute code ;-)
838             */
839
840        case LSPPING_TLV_PAD:
841        case LSPPING_TLV_ERROR_CODE:
842        case LSPPING_TLV_VENDOR_PRIVATE:
843
844        default:
845            if (vflag <= 1)
846                print_unknown_data(tlv_tptr,"\n\t    ",tlv_tlen);
847            break;
848        }
849        /* do we want to see an additionally tlv hexdump ? */
850        if (vflag > 1 || tlv_hexdump==TRUE)
851            print_unknown_data(tptr+sizeof(sizeof(struct lspping_tlv_header)),"\n\t    ",
852                               lspping_tlv_len);
853
854        tptr+=lspping_tlv_len+sizeof(struct lspping_tlv_header);
855        tlen-=lspping_tlv_len+sizeof(struct lspping_tlv_header);
856    }
857    return;
858trunc:
859    printf("\n\t\t packet exceeded snapshot");
860}
861