tcpdump/print-pim.c

1235 lines
33 KiB
C
Raw Normal View History

1999-10-08 07:47:09 +08:00
/*
* Copyright (c) 1995, 1996
* The Regents of the University of California. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that: (1) source code distributions
* retain the above copyright notice and this paragraph in its entirety, (2)
* distributions including binary code include the above copyright notice and
* this paragraph in its entirety in the documentation or other materials
* provided with the distribution, and (3) all advertising materials mentioning
* features or use of this software display the following acknowledgement:
* ``This product includes software developed by the University of California,
* Lawrence Berkeley Laboratory and its contributors.'' Neither the name of
* the University nor the names of its contributors may be used to endorse
* or promote products derived from this software without specific prior
* written permission.
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
* WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
*/
/* \summary: Protocol Independent Multicast (PIM) printer */
#ifdef HAVE_CONFIG_H
#include <config.h>
1999-10-08 07:47:09 +08:00
#endif
#include "netdissect-stdinc.h"
#include "netdissect.h"
#include "addrtoname.h"
#include "extract.h"
#include "ip.h"
#include "ip6.h"
#include "ipproto.h"
#define PIMV1_TYPE_QUERY 0
#define PIMV1_TYPE_REGISTER 1
#define PIMV1_TYPE_REGISTER_STOP 2
#define PIMV1_TYPE_JOIN_PRUNE 3
#define PIMV1_TYPE_RP_REACHABILITY 4
#define PIMV1_TYPE_ASSERT 5
#define PIMV1_TYPE_GRAFT 6
#define PIMV1_TYPE_GRAFT_ACK 7
static const struct tok pimv1_type_str[] = {
{ PIMV1_TYPE_QUERY, "Query" },
{ PIMV1_TYPE_REGISTER, "Register" },
{ PIMV1_TYPE_REGISTER_STOP, "Register-Stop" },
{ PIMV1_TYPE_JOIN_PRUNE, "Join/Prune" },
{ PIMV1_TYPE_RP_REACHABILITY, "RP-reachable" },
{ PIMV1_TYPE_ASSERT, "Assert" },
{ PIMV1_TYPE_GRAFT, "Graft" },
{ PIMV1_TYPE_GRAFT_ACK, "Graft-ACK" },
{ 0, NULL }
};
#define PIMV2_TYPE_HELLO 0
#define PIMV2_TYPE_REGISTER 1
#define PIMV2_TYPE_REGISTER_STOP 2
#define PIMV2_TYPE_JOIN_PRUNE 3
#define PIMV2_TYPE_BOOTSTRAP 4
#define PIMV2_TYPE_ASSERT 5
#define PIMV2_TYPE_GRAFT 6
#define PIMV2_TYPE_GRAFT_ACK 7
#define PIMV2_TYPE_CANDIDATE_RP 8
#define PIMV2_TYPE_PRUNE_REFRESH 9
#define PIMV2_TYPE_DF_ELECTION 10
#define PIMV2_TYPE_ECMP_REDIRECT 11
static const struct tok pimv2_type_values[] = {
{ PIMV2_TYPE_HELLO, "Hello" },
{ PIMV2_TYPE_REGISTER, "Register" },
{ PIMV2_TYPE_REGISTER_STOP, "Register Stop" },
{ PIMV2_TYPE_JOIN_PRUNE, "Join / Prune" },
{ PIMV2_TYPE_BOOTSTRAP, "Bootstrap" },
{ PIMV2_TYPE_ASSERT, "Assert" },
{ PIMV2_TYPE_GRAFT, "Graft" },
{ PIMV2_TYPE_GRAFT_ACK, "Graft Acknowledgement" },
{ PIMV2_TYPE_CANDIDATE_RP, "Candidate RP Advertisement" },
{ PIMV2_TYPE_PRUNE_REFRESH, "Prune Refresh" },
{ PIMV2_TYPE_DF_ELECTION, "DF Election" },
{ PIMV2_TYPE_ECMP_REDIRECT, "ECMP Redirect" },
{ 0, NULL}
};
#define PIMV2_HELLO_OPTION_HOLDTIME 1
#define PIMV2_HELLO_OPTION_LANPRUNEDELAY 2
#define PIMV2_HELLO_OPTION_DR_PRIORITY_OLD 18
#define PIMV2_HELLO_OPTION_DR_PRIORITY 19
#define PIMV2_HELLO_OPTION_GENID 20
#define PIMV2_HELLO_OPTION_REFRESH_CAP 21
#define PIMV2_HELLO_OPTION_BIDIR_CAP 22
#define PIMV2_HELLO_OPTION_ADDRESS_LIST 24
#define PIMV2_HELLO_OPTION_ADDRESS_LIST_OLD 65001
static const struct tok pimv2_hello_option_values[] = {
{ PIMV2_HELLO_OPTION_HOLDTIME, "Hold Time" },
{ PIMV2_HELLO_OPTION_LANPRUNEDELAY, "LAN Prune Delay" },
{ PIMV2_HELLO_OPTION_DR_PRIORITY_OLD, "DR Priority (Old)" },
{ PIMV2_HELLO_OPTION_DR_PRIORITY, "DR Priority" },
{ PIMV2_HELLO_OPTION_GENID, "Generation ID" },
{ PIMV2_HELLO_OPTION_REFRESH_CAP, "State Refresh Capability" },
{ PIMV2_HELLO_OPTION_BIDIR_CAP, "Bi-Directional Capability" },
{ PIMV2_HELLO_OPTION_ADDRESS_LIST, "Address List" },
{ PIMV2_HELLO_OPTION_ADDRESS_LIST_OLD, "Address List (Old)" },
{ 0, NULL}
};
#define PIMV2_REGISTER_FLAG_LEN 4
#define PIMV2_REGISTER_FLAG_BORDER 0x80000000
#define PIMV2_REGISTER_FLAG_NULL 0x40000000
static const struct tok pimv2_register_flag_values[] = {
{ PIMV2_REGISTER_FLAG_BORDER, "Border" },
{ PIMV2_REGISTER_FLAG_NULL, "Null" },
{ 0, NULL}
2014-01-02 10:27:54 +08:00
};
1999-10-08 07:47:09 +08:00
#define PIMV2_DF_ELECTION_OFFER 1
#define PIMV2_DF_ELECTION_WINNER 2
#define PIMV2_DF_ELECTION_BACKOFF 3
#define PIMV2_DF_ELECTION_PASS 4
static const struct tok pimv2_df_election_flag_values[] = {
{ PIMV2_DF_ELECTION_OFFER, "Offer" },
{ PIMV2_DF_ELECTION_WINNER, "Winner" },
{ PIMV2_DF_ELECTION_BACKOFF, "Backoff" },
{ PIMV2_DF_ELECTION_PASS, "Pass" },
{ 0, NULL}
};
#define PIMV2_DF_ELECTION_PASS_BACKOFF_STR(x) ( \
x == PIMV2_DF_ELECTION_BACKOFF ? "offer" : "new winner" )
/*
* XXX: We consider a case where IPv6 is not ready yet for portability,
2020-12-12 23:10:25 +08:00
* but PIM dependent definitions should be independent of IPv6...
*/
struct pim {
nd_uint8_t pim_typever;
/* upper 4bit: PIM version number; 2 for PIMv2 */
/* lower 4bit: the PIM message type, currently they are:
* Hello, Register, Register-Stop, Join/Prune,
* Bootstrap, Assert, Graft (PIM-DM only),
* Graft-Ack (PIM-DM only), C-RP-Adv
*/
#define PIM_VER(x) (((x) & 0xf0) >> 4)
#define PIM_TYPE(x) ((x) & 0x0f)
nd_uint8_t pim_rsv; /* Reserved in v1, subtype+address length in v2 */
#define PIM_SUBTYPE(x) (((x) & 0xf0) >> 4)
nd_uint16_t pim_cksum; /* IP style check sum */
};
static void pimv2_print(netdissect_options *, const u_char *bp, u_int len, const u_char *);
static void
pimv1_join_prune_print(netdissect_options *ndo,
const u_char *bp, u_int len)
{
u_int ngroups, njoin, nprune;
u_int njp;
2001-01-28 16:27:28 +08:00
/* If it's a single group and a single source, use 1-line output. */
if (ND_TTEST_LEN(bp, 30) && GET_U_1(bp + 11) == 1 &&
((njoin = GET_BE_U_2(bp + 20)) + GET_BE_U_2(bp + 22)) == 1) {
u_int hold;
2001-01-28 16:27:28 +08:00
ND_PRINT(" RPF %s ", GET_IPADDR_STRING(bp));
hold = GET_BE_U_2(bp + 6);
2001-01-28 16:27:28 +08:00
if (hold != 180) {
2018-01-07 18:47:30 +08:00
ND_PRINT("Hold ");
unsigned_relts_print(ndo, hold);
2001-01-28 16:27:28 +08:00
}
2018-01-07 18:47:30 +08:00
ND_PRINT("%s (%s/%u, %s", njoin ? "Join" : "Prune",
GET_IPADDR_STRING(bp + 26), GET_U_1(bp + 25) & 0x3f,
GET_IPADDR_STRING(bp + 12));
if (GET_BE_U_4(bp + 16) != 0xffffffff)
ND_PRINT("/%s", GET_IPADDR_STRING(bp + 16));
2018-01-07 18:47:30 +08:00
ND_PRINT(") %s%s %s",
(GET_U_1(bp + 24) & 0x01) ? "Sparse" : "Dense",
(GET_U_1(bp + 25) & 0x80) ? " WC" : "",
(GET_U_1(bp + 25) & 0x40) ? "RP" : "SPT");
2001-01-28 16:27:28 +08:00
return;
}
if (len < sizeof(nd_ipv4))
goto trunc;
if (ndo->ndo_vflag > 1)
2018-01-07 18:47:30 +08:00
ND_PRINT("\n");
ND_PRINT(" Upstream Nbr: %s", GET_IPADDR_STRING(bp));
bp += 4;
len -= 4;
if (len < 4)
goto trunc;
if (ndo->ndo_vflag > 1)
2018-01-07 18:47:30 +08:00
ND_PRINT("\n");
ND_PRINT(" Hold time: ");
unsigned_relts_print(ndo, GET_BE_U_2(bp + 2));
if (ndo->ndo_vflag < 2)
return;
bp += 4;
len -= 4;
2001-01-28 16:27:28 +08:00
if (len < 4)
goto trunc;
ngroups = GET_U_1(bp + 3);
2001-01-28 16:27:28 +08:00
bp += 4;
len -= 4;
while (ngroups != 0) {
/*
* XXX - does the address have length "addrlen" and the
* mask length "maddrlen"?
*/
if (len < 4)
goto trunc;
ND_PRINT("\n\tGroup: %s", GET_IPADDR_STRING(bp));
bp += 4;
len -= 4;
if (len < 4)
goto trunc;
if (GET_BE_U_4(bp) != 0xffffffff)
ND_PRINT("/%s", GET_IPADDR_STRING(bp));
bp += 4;
len -= 4;
if (len < 4)
goto trunc;
njoin = GET_BE_U_2(bp);
nprune = GET_BE_U_2(bp + 2);
2018-01-07 18:47:30 +08:00
ND_PRINT(" joined: %u pruned: %u", njoin, nprune);
bp += 4;
len -= 4;
2001-01-28 16:27:28 +08:00
for (njp = 0; njp < (njoin + nprune); njp++) {
Add a few more GCC warnings on GCC >= 2 for ".devel" builds. From Neil T. Spring: fixes for many of those warnings: addrtoname.c, configure.in: Linux needs netinet/ether.h for ether_ntohost print-*.c: change char *foo = "bar" to const char *foo = "bar" to appease -Wwrite-strings; should affect no run-time behavior. print-*.c: make some variables unsigned. print-bgp.c: plen ('prefix len') is unsigned, no reason to validate by comparing to zero. print-cnfp.c, print-rx.c: use intoa, provided by addrtoname, instead of inet_ntoa. print-domain.c: unsigned int l; (l=foo()) < 0 is guaranteed to be false, so check for (u_int)-1, which represents failure, explicitly. print-isakmp.c: complete initialization of attrmap objects. print-lwres.c: "if(x); print foo;" seemed much more likely to be intended to be "if(x) { print foo; }". print-smb.c: complete initialization of some structures. In addition, add some fixes for the signed vs. unsigned comparison warnings: extract.h: cast the result of the byte-extraction-and-combining, as, at least for the 16-bit version, C's integral promotions will turn "u_int16_t" into "int" if there are other "int"s nearby. print-*.c: make some more variables unsigned, or add casts to an unsigned type of signed values known not to be negative, or add casts to "int" of unsigned values known to fit in an "int", and make other changes needed to handle the aforementioned variables now being unsigned. print-isakmp.c: clean up the handling of error/status indicators in notify messages. print-ppp.c: get rid of a check that an unsigned quantity is >= 0. print-radius.c: clean up some of the bounds checking. print-smb.c: extract the word count into a "u_int" to avoid the aforementioned problems with C's integral promotions. print-snmp.c: change a check that an unsigned variable is >= 0 to a check that it's != 0. Also, fix some formats to use "%u" rather than "%d" for unsigned quantities.
2002-09-05 08:00:07 +08:00
const char *type;
2001-01-28 16:27:28 +08:00
if (njp < njoin)
type = "Join ";
else
type = "Prune";
if (len < 6)
goto trunc;
2018-01-07 18:47:30 +08:00
ND_PRINT("\n\t%s %s%s%s%s/%u", type,
(GET_U_1(bp) & 0x01) ? "Sparse " : "Dense ",
(GET_U_1(bp + 1) & 0x80) ? "WC " : "",
(GET_U_1(bp + 1) & 0x40) ? "RP " : "SPT ",
GET_IPADDR_STRING(bp + 2),
GET_U_1(bp + 1) & 0x3f);
2001-01-28 16:27:28 +08:00
bp += 6;
len -= 6;
}
ngroups--;
}
2001-01-28 16:27:28 +08:00
return;
trunc:
nd_print_trunc(ndo);
}
1999-10-08 07:47:09 +08:00
void
pimv1_print(netdissect_options *ndo,
const u_char *bp, u_int len)
1999-10-08 07:47:09 +08:00
{
u_char type;
1999-10-08 07:47:09 +08:00
ndo->ndo_protocol = "pimv1";
type = GET_U_1(bp + 1);
1999-10-08 07:47:09 +08:00
2018-01-07 18:47:30 +08:00
ND_PRINT(" %s", tok2str(pimv1_type_str, "[type %u]", type));
2001-01-28 16:27:28 +08:00
switch (type) {
case PIMV1_TYPE_QUERY:
2017-12-16 02:39:00 +08:00
if (ND_TTEST_1(bp + 8)) {
switch (GET_U_1(bp + 8) >> 4) {
2001-01-28 16:27:28 +08:00
case 0:
2018-01-07 18:47:30 +08:00
ND_PRINT(" Dense-mode");
break;
2001-01-28 16:27:28 +08:00
case 1:
2018-01-07 18:47:30 +08:00
ND_PRINT(" Sparse-mode");
break;
2001-01-28 16:27:28 +08:00
case 2:
2018-01-07 18:47:30 +08:00
ND_PRINT(" Sparse-Dense-mode");
break;
2001-01-28 16:27:28 +08:00
default:
ND_PRINT(" mode-%u", GET_U_1(bp + 8) >> 4);
break;
2001-01-28 16:27:28 +08:00
}
}
if (ndo->ndo_vflag) {
2018-01-07 18:47:30 +08:00
ND_PRINT(" (Hold-time ");
unsigned_relts_print(ndo, GET_BE_U_2(bp + 10));
2018-01-07 18:47:30 +08:00
ND_PRINT(")");
2001-01-28 16:27:28 +08:00
}
break;
1999-10-08 07:47:09 +08:00
case PIMV1_TYPE_REGISTER:
ND_TCHECK_LEN(bp + 8, 20); /* ip header */
ND_PRINT(" for %s > %s", GET_IPADDR_STRING(bp + 20),
GET_IPADDR_STRING(bp + 24));
2001-01-28 16:27:28 +08:00
break;
case PIMV1_TYPE_REGISTER_STOP:
ND_PRINT(" for %s > %s", GET_IPADDR_STRING(bp + 8),
GET_IPADDR_STRING(bp + 12));
2001-01-28 16:27:28 +08:00
break;
case PIMV1_TYPE_RP_REACHABILITY:
if (ndo->ndo_vflag) {
ND_PRINT(" group %s", GET_IPADDR_STRING(bp + 8));
if (GET_BE_U_4(bp + 12) != 0xffffffff)
ND_PRINT("/%s", GET_IPADDR_STRING(bp + 12));
ND_PRINT(" RP %s hold ", GET_IPADDR_STRING(bp + 16));
unsigned_relts_print(ndo, GET_BE_U_2(bp + 22));
2001-01-28 16:27:28 +08:00
}
break;
case PIMV1_TYPE_ASSERT:
ND_PRINT(" for %s > %s", GET_IPADDR_STRING(bp + 16),
GET_IPADDR_STRING(bp + 8));
if (GET_BE_U_4(bp + 12) != 0xffffffff)
ND_PRINT("/%s", GET_IPADDR_STRING(bp + 12));
2018-01-07 18:47:30 +08:00
ND_PRINT(" %s pref %u metric %u",
(GET_U_1(bp + 20) & 0x80) ? "RP-tree" : "SPT",
GET_BE_U_4(bp + 20) & 0x7fffffff,
GET_BE_U_4(bp + 24));
2001-01-28 16:27:28 +08:00
break;
case PIMV1_TYPE_JOIN_PRUNE:
case PIMV1_TYPE_GRAFT:
case PIMV1_TYPE_GRAFT_ACK:
if (ndo->ndo_vflag) {
if (len < 8)
goto trunc;
pimv1_join_prune_print(ndo, bp + 8, len - 8);
}
2001-01-28 16:27:28 +08:00
break;
}
if ((GET_U_1(bp + 4) >> 4) != 1)
ND_PRINT(" [v%u]", GET_U_1(bp + 4) >> 4);
2001-01-28 16:27:28 +08:00
return;
trunc:
nd_print_trunc(ndo);
}
/*
* auto-RP is a cisco protocol, documented at
* ftp://ftpeng.cisco.com/ipmulticast/specs/pim-autorp-spec01.txt
*
* This implements version 1+, dated Sept 9, 1998.
*/
void
cisco_autorp_print(netdissect_options *ndo,
const u_char *bp, u_int len)
{
u_int type;
u_int numrps;
u_int hold;
ndo->ndo_protocol = "cisco_autorp";
if (len < 8)
goto trunc;
2018-01-07 18:47:30 +08:00
ND_PRINT(" auto-rp ");
type = GET_U_1(bp);
2001-01-28 16:27:28 +08:00
switch (type) {
case 0x11:
2018-01-07 18:47:30 +08:00
ND_PRINT("candidate-advert");
break;
2001-01-28 16:27:28 +08:00
case 0x12:
2018-01-07 18:47:30 +08:00
ND_PRINT("mapping");
break;
2001-01-28 16:27:28 +08:00
default:
2018-01-07 18:47:30 +08:00
ND_PRINT("type-0x%02x", type);
break;
}
2001-01-28 16:27:28 +08:00
numrps = GET_U_1(bp + 1);
2001-01-28 16:27:28 +08:00
2018-01-07 18:47:30 +08:00
ND_PRINT(" Hold ");
hold = GET_BE_U_2(bp + 2);
2001-01-28 16:27:28 +08:00
if (hold)
unsigned_relts_print(ndo, GET_BE_U_2(bp + 2));
2001-01-28 16:27:28 +08:00
else
2018-01-07 18:47:30 +08:00
ND_PRINT("FOREVER");
2001-01-28 16:27:28 +08:00
/* Next 4 bytes are reserved. */
bp += 8; len -= 8;
/*XXX skip unless -v? */
/*
* Rest of packet:
* numrps entries of the form:
* 32 bits: RP
* 6 bits: reserved
* 2 bits: PIM version supported, bit 0 is "supports v1", 1 is "v2".
* 8 bits: # of entries for this RP
* each entry: 7 bits: reserved, 1 bit: negative,
* 8 bits: mask 32 bits: source
* lather, rinse, repeat.
*/
while (numrps != 0) {
u_int nentries;
2001-01-28 16:27:28 +08:00
char s;
if (len < 4)
goto trunc;
ND_PRINT(" RP %s", GET_IPADDR_STRING(bp));
bp += 4;
len -= 4;
if (len < 1)
goto trunc;
switch (GET_U_1(bp) & 0x3) {
2018-01-07 18:47:30 +08:00
case 0: ND_PRINT(" PIMv?");
2001-01-28 16:27:28 +08:00
break;
2018-01-07 18:47:30 +08:00
case 1: ND_PRINT(" PIMv1");
2001-01-28 16:27:28 +08:00
break;
2018-01-07 18:47:30 +08:00
case 2: ND_PRINT(" PIMv2");
2001-01-28 16:27:28 +08:00
break;
2018-01-07 18:47:30 +08:00
case 3: ND_PRINT(" PIMv1+2");
2001-01-28 16:27:28 +08:00
break;
}
if (GET_U_1(bp) & 0xfc)
ND_PRINT(" [rsvd=0x%02x]", GET_U_1(bp) & 0xfc);
bp += 1;
len -= 1;
if (len < 1)
goto trunc;
nentries = GET_U_1(bp);
bp += 1;
len -= 1;
2001-01-28 16:27:28 +08:00
s = ' ';
while (nentries != 0) {
if (len < 6)
goto trunc;
ND_PRINT("%c%s%s/%u", s, GET_U_1(bp) & 1 ? "!" : "",
GET_IPADDR_STRING(bp + 2), GET_U_1(bp + 1));
if (GET_U_1(bp) & 0x02) {
2018-01-07 18:47:30 +08:00
ND_PRINT(" bidir");
}
if (GET_U_1(bp) & 0xfc) {
ND_PRINT("[rsvd=0x%02x]", GET_U_1(bp) & 0xfc);
}
2001-01-28 16:27:28 +08:00
s = ',';
bp += 6; len -= 6;
nentries--;
2001-01-28 16:27:28 +08:00
}
numrps--;
}
2001-01-28 16:27:28 +08:00
return;
trunc:
nd_print_trunc(ndo);
1999-10-08 07:47:09 +08:00
}
void
pim_print(netdissect_options *ndo,
const u_char *bp, u_int len, const u_char *bp2)
{
const struct pim *pim = (const struct pim *)bp;
uint8_t pim_typever;
ndo->ndo_protocol = "pim";
pim_typever = GET_U_1(pim->pim_typever);
switch (PIM_VER(pim_typever)) {
case 2:
if (!ndo->ndo_vflag) {
2018-01-07 18:47:30 +08:00
ND_PRINT("PIMv%u, %s, length %u",
PIM_VER(pim_typever),
tok2str(pimv2_type_values,"Unknown Type",PIM_TYPE(pim_typever)),
2018-01-07 18:47:30 +08:00
len);
return;
} else {
2018-01-07 18:47:30 +08:00
ND_PRINT("PIMv%u, length %u\n\t%s",
PIM_VER(pim_typever),
len,
2018-01-07 18:47:30 +08:00
tok2str(pimv2_type_values,"Unknown Type",PIM_TYPE(pim_typever)));
pimv2_print(ndo, bp, len, bp2);
}
break;
2001-01-28 16:27:28 +08:00
default:
2018-01-07 18:47:30 +08:00
ND_PRINT("PIMv%u, length %u",
PIM_VER(pim_typever),
2018-01-07 18:47:30 +08:00
len);
break;
}
}
/*
* PIMv2 uses encoded address representations.
*
* The last PIM-SM I-D before RFC2117 was published specified the
* following representation for unicast addresses. However, RFC2117
* specified no encoding for unicast addresses with the unicast
* address length specified in the header. Therefore, we have to
* guess which encoding is being used (Cisco's PIMv2 implementation
* uses the non-RFC encoding). RFC2117 turns a previously "Reserved"
* field into a 'unicast-address-length-in-bytes' field. We guess
* that it's the draft encoding if this reserved field is zero.
*
* RFC2362 goes back to the encoded format, and calls the addr length
* field "reserved" again.
*
* The first byte is the address family, from:
*
* 0 Reserved
* 1 IP (IP version 4)
* 2 IP6 (IP version 6)
* 3 NSAP
* 4 HDLC (8-bit multidrop)
* 5 BBN 1822
* 6 802 (includes all 802 media plus Ethernet "canonical format")
* 7 E.163
* 8 E.164 (SMDS, Frame Relay, ATM)
* 9 F.69 (Telex)
* 10 X.121 (X.25, Frame Relay)
* 11 IPX
* 12 Appletalk
* 13 Decnet IV
* 14 Banyan Vines
* 15 E.164 with NSAP format subaddress
*
* In addition, the second byte is an "Encoding". 0 is the default
* encoding for the address family, and no other encodings are currently
* specified.
*
*/
enum pimv2_addrtype {
pimv2_unicast, pimv2_group, pimv2_source
};
/* 0 1 2 3
* 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
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | Addr Family | Encoding Type | Unicast Address |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+++++++
* 0 1 2 3
* 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
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | Addr Family | Encoding Type | Reserved | Mask Len |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | Group multicast Address |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* 0 1 2 3
* 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
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | Addr Family | Encoding Type | Rsrvd |S|W|R| Mask Len |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | Source Address |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
*/
static int
pimv2_addr_print(netdissect_options *ndo,
const u_char *bp, u_int len, enum pimv2_addrtype at,
u_int addr_len, int silent)
{
u_int af;
int hdrlen;
if (addr_len == 0) {
if (len < 2)
goto trunc;
switch (GET_U_1(bp)) {
2001-01-28 16:27:28 +08:00
case 1:
af = AF_INET;
addr_len = (u_int)sizeof(nd_ipv4);
break;
2001-01-28 16:27:28 +08:00
case 2:
af = AF_INET6;
addr_len = (u_int)sizeof(nd_ipv6);
break;
2001-01-28 16:27:28 +08:00
default:
return -1;
}
if (GET_U_1(bp + 1) != 0)
return -1;
hdrlen = 2;
} else {
switch (addr_len) {
case sizeof(nd_ipv4):
af = AF_INET;
break;
case sizeof(nd_ipv6):
af = AF_INET6;
break;
2001-01-28 16:27:28 +08:00
default:
return -1;
break;
}
hdrlen = 0;
}
bp += hdrlen;
len -= hdrlen;
switch (at) {
2001-01-28 16:27:28 +08:00
case pimv2_unicast:
if (len < addr_len)
goto trunc;
ND_TCHECK_LEN(bp, addr_len);
if (af == AF_INET) {
if (!silent)
ND_PRINT("%s", GET_IPADDR_STRING(bp));
}
else if (af == AF_INET6) {
if (!silent)
ND_PRINT("%s", GET_IP6ADDR_STRING(bp));
}
return hdrlen + addr_len;
2001-01-28 16:27:28 +08:00
case pimv2_group:
case pimv2_source:
if (len < addr_len + 2)
goto trunc;
ND_TCHECK_LEN(bp, addr_len + 2);
if (af == AF_INET) {
if (!silent) {
ND_PRINT("%s", GET_IPADDR_STRING(bp + 2));
if (GET_U_1(bp + 1) != 32)
ND_PRINT("/%u", GET_U_1(bp + 1));
}
}
else if (af == AF_INET6) {
if (!silent) {
ND_PRINT("%s", GET_IP6ADDR_STRING(bp + 2));
if (GET_U_1(bp + 1) != 128)
ND_PRINT("/%u", GET_U_1(bp + 1));
}
}
if (GET_U_1(bp) && !silent) {
if (at == pimv2_group) {
ND_PRINT("(0x%02x)", GET_U_1(bp));
} else {
2018-01-07 18:47:30 +08:00
ND_PRINT("(%s%s%s",
GET_U_1(bp) & 0x04 ? "S" : "",
GET_U_1(bp) & 0x02 ? "W" : "",
GET_U_1(bp) & 0x01 ? "R" : "");
if (GET_U_1(bp) & 0xf8) {
ND_PRINT("+0x%02x",
GET_U_1(bp) & 0xf8);
}
2018-01-07 18:47:30 +08:00
ND_PRINT(")");
}
}
return hdrlen + 2 + addr_len;
default:
return -1;
}
trunc:
return -1;
}
enum checksum_status {
CORRECT,
INCORRECT,
UNVERIFIED
};
static enum checksum_status
pimv2_check_checksum(netdissect_options *ndo, const u_char *bp,
const u_char *bp2, u_int len)
{
const struct ip *ip;
u_int cksum;
if (!ND_TTEST_LEN(bp, len)) {
/* We don't have all the data. */
return (UNVERIFIED);
}
ip = (const struct ip *)bp2;
if (IP_V(ip) == 4) {
struct cksum_vec vec[1];
vec[0].ptr = bp;
vec[0].len = len;
cksum = in_cksum(vec, 1);
return (cksum ? INCORRECT : CORRECT);
} else if (IP_V(ip) == 6) {
const struct ip6_hdr *ip6;
ip6 = (const struct ip6_hdr *)bp2;
cksum = nextproto6_cksum(ndo, ip6, bp, len, len, IPPROTO_PIM);
return (cksum ? INCORRECT : CORRECT);
} else {
return (UNVERIFIED);
}
}
static void
pimv2_print(netdissect_options *ndo,
const u_char *bp, u_int len, const u_char *bp2)
{
const struct pim *pim = (const struct pim *)bp;
int advance;
int subtype;
enum checksum_status cksum_status;
u_int pim_typever;
u_int pimv2_addr_len;
ndo->ndo_protocol = "pimv2";
2019-11-09 17:39:53 +08:00
if (len < 2) {
ND_PRINT("[length %u < 2]", len);
nd_print_invalid(ndo);
return;
}
pim_typever = GET_U_1(pim->pim_typever);
/* RFC5015 allocates the high 4 bits of pim_rsv for "subtype". */
pimv2_addr_len = GET_U_1(pim->pim_rsv) & 0x0f;
if (pimv2_addr_len != 0)
2018-01-07 18:47:30 +08:00
ND_PRINT(", RFC2117-encoding");
2019-11-09 17:39:53 +08:00
if (len < 4) {
ND_PRINT("[length %u < 4]", len);
nd_print_invalid(ndo);
return;
}
ND_PRINT(", cksum 0x%04x ", GET_BE_U_2(pim->pim_cksum));
if (GET_BE_U_2(pim->pim_cksum) == 0) {
2018-01-07 18:47:30 +08:00
ND_PRINT("(unverified)");
} else {
if (PIM_TYPE(pim_typever) == PIMV2_TYPE_REGISTER) {
/*
* The checksum only covers the packet header,
* not the encapsulated packet.
*/
cksum_status = pimv2_check_checksum(ndo, bp, bp2, 8);
if (cksum_status == INCORRECT) {
/*
* To quote RFC 4601, "For interoperability
* reasons, a message carrying a checksum
* calculated over the entire PIM Register
* message should also be accepted."
*/
cksum_status = pimv2_check_checksum(ndo, bp, bp2, len);
}
} else {
/*
* The checksum covers the entire packet.
*/
cksum_status = pimv2_check_checksum(ndo, bp, bp2, len);
}
switch (cksum_status) {
case CORRECT:
2018-01-07 18:47:30 +08:00
ND_PRINT("(correct)");
break;
case INCORRECT:
2018-01-07 18:47:30 +08:00
ND_PRINT("(incorrect)");
break;
case UNVERIFIED:
2018-01-07 18:47:30 +08:00
ND_PRINT("(unverified)");
break;
}
}
bp += 4;
len -= 4;
switch (PIM_TYPE(pim_typever)) {
case PIMV2_TYPE_HELLO:
{
uint16_t otype, olen;
while (len > 0) {
if (len < 4)
goto trunc;
otype = GET_BE_U_2(bp);
olen = GET_BE_U_2(bp + 2);
2018-01-07 18:47:30 +08:00
ND_PRINT("\n\t %s Option (%u), length %u, Value: ",
tok2str(pimv2_hello_option_values, "Unknown", otype),
otype,
2018-01-07 18:47:30 +08:00
olen);
bp += 4;
len -= 4;
if (len < olen)
goto trunc;
ND_TCHECK_LEN(bp, olen);
switch (otype) {
case PIMV2_HELLO_OPTION_HOLDTIME:
if (olen != 2) {
ND_PRINT("[option length %u != 2]", olen);
nd_print_invalid(ndo);
return;
} else {
unsigned_relts_print(ndo,
GET_BE_U_2(bp));
}
break;
case PIMV2_HELLO_OPTION_LANPRUNEDELAY:
if (olen != 4) {
ND_PRINT("[option length %u != 4]", olen);
nd_print_invalid(ndo);
return;
} else {
char t_bit;
uint16_t lan_delay, override_interval;
lan_delay = GET_BE_U_2(bp);
override_interval = GET_BE_U_2(bp + 2);
t_bit = (lan_delay & 0x8000)? 1 : 0;
lan_delay &= ~0x8000;
2018-01-07 18:47:30 +08:00
ND_PRINT("\n\t T-bit=%u, LAN delay %ums, Override interval %ums",
t_bit, lan_delay, override_interval);
}
break;
case PIMV2_HELLO_OPTION_DR_PRIORITY_OLD:
case PIMV2_HELLO_OPTION_DR_PRIORITY:
switch (olen) {
case 0:
2018-01-07 18:47:30 +08:00
ND_PRINT("Bi-Directional Capability (Old)");
break;
case 4:
ND_PRINT("%u", GET_BE_U_4(bp));
break;
default:
ND_PRINT("[option length %u != 4]", olen);
nd_print_invalid(ndo);
return;
break;
}
break;
case PIMV2_HELLO_OPTION_GENID:
if (olen != 4) {
ND_PRINT("[option length %u != 4]", olen);
nd_print_invalid(ndo);
return;
} else {
ND_PRINT("0x%08x", GET_BE_U_4(bp));
}
break;
case PIMV2_HELLO_OPTION_REFRESH_CAP:
if (olen != 4) {
ND_PRINT("[option length %u != 4]", olen);
nd_print_invalid(ndo);
return;
} else {
ND_PRINT("v%u", GET_U_1(bp));
if (GET_U_1(bp + 1) != 0) {
2018-01-07 18:47:30 +08:00
ND_PRINT(", interval ");
unsigned_relts_print(ndo,
GET_U_1(bp + 1));
}
if (GET_BE_U_2(bp + 2) != 0) {
ND_PRINT(" ?0x%04x?",
GET_BE_U_2(bp + 2));
}
}
break;
case PIMV2_HELLO_OPTION_BIDIR_CAP:
break;
case PIMV2_HELLO_OPTION_ADDRESS_LIST_OLD:
case PIMV2_HELLO_OPTION_ADDRESS_LIST:
if (ndo->ndo_vflag > 1) {
const u_char *ptr = bp;
u_int plen = len;
while (ptr < (bp+olen)) {
2018-01-07 18:47:30 +08:00
ND_PRINT("\n\t ");
advance = pimv2_addr_print(ndo, ptr, plen, pimv2_unicast, pimv2_addr_len, 0);
if (advance < 0)
goto trunc;
ptr += advance;
plen -= advance;
}
}
break;
default:
if (ndo->ndo_vflag <= 1)
print_unknown_data(ndo, bp, "\n\t ", olen);
break;
}
/* do we want to see an additionally hexdump ? */
if (ndo->ndo_vflag> 1)
print_unknown_data(ndo, bp, "\n\t ", olen);
bp += olen;
len -= olen;
}
break;
}
case PIMV2_TYPE_REGISTER:
2001-01-28 16:27:28 +08:00
{
2015-04-27 08:24:42 +08:00
const struct ip *ip;
if (len < 4)
goto trunc;
ND_TCHECK_LEN(bp, PIMV2_REGISTER_FLAG_LEN);
2018-01-07 18:47:30 +08:00
ND_PRINT(", Flags [ %s ]\n\t",
tok2str(pimv2_register_flag_values,
"none",
GET_BE_U_4(bp)));
bp += 4; len -= 4;
/* encapsulated multicast packet */
if (len == 0)
goto trunc;
2015-04-27 08:24:42 +08:00
ip = (const struct ip *)bp;
switch (IP_V(ip)) {
case 0: /* Null header */
2018-01-07 18:47:30 +08:00
ND_PRINT("IP-Null-header %s > %s",
GET_IPADDR_STRING(ip->ip_src),
GET_IPADDR_STRING(ip->ip_dst));
break;
2001-01-28 16:27:28 +08:00
case 4: /* IPv4 */
ip_print(ndo, bp, len);
break;
2001-01-28 16:27:28 +08:00
case 6: /* IPv6 */
ip6_print(ndo, bp, len);
break;
default:
2018-01-07 18:47:30 +08:00
ND_PRINT("IP ver %u", IP_V(ip));
break;
}
break;
2001-01-28 16:27:28 +08:00
}
case PIMV2_TYPE_REGISTER_STOP:
2018-01-07 18:47:30 +08:00
ND_PRINT(" group=");
if ((advance = pimv2_addr_print(ndo, bp, len, pimv2_group, pimv2_addr_len, 0)) < 0)
goto trunc;
bp += advance; len -= advance;
2018-01-07 18:47:30 +08:00
ND_PRINT(" source=");
if ((advance = pimv2_addr_print(ndo, bp, len, pimv2_unicast, pimv2_addr_len, 0)) < 0)
goto trunc;
bp += advance; len -= advance;
break;
case PIMV2_TYPE_JOIN_PRUNE:
case PIMV2_TYPE_GRAFT:
case PIMV2_TYPE_GRAFT_ACK:
/*
* 0 1 2 3
* 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
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* |PIM Ver| Type | Addr length | Checksum |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | Unicast-Upstream Neighbor Address |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | Reserved | Num groups | Holdtime |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | Encoded-Multicast Group Address-1 |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | Number of Joined Sources | Number of Pruned Sources |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | Encoded-Joined Source Address-1 |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | . |
* | . |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | Encoded-Joined Source Address-n |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | Encoded-Pruned Source Address-1 |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | . |
* | . |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | Encoded-Pruned Source Address-n |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | . |
* | . |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | Encoded-Multicast Group Address-n |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
*/
{
uint8_t ngroup;
uint16_t holdtime;
uint16_t njoin;
uint16_t nprune;
u_int i, j;
if (PIM_TYPE(pim_typever) != 7) { /*not for Graft-ACK*/
2018-01-07 18:47:30 +08:00
ND_PRINT(", upstream-neighbor: ");
if ((advance = pimv2_addr_print(ndo, bp, len, pimv2_unicast, pimv2_addr_len, 0)) < 0)
goto trunc;
bp += advance; len -= advance;
}
if (len < 4)
goto trunc;
ND_TCHECK_4(bp);
ngroup = GET_U_1(bp + 1);
holdtime = GET_BE_U_2(bp + 2);
2018-01-07 18:47:30 +08:00
ND_PRINT("\n\t %u group(s)", ngroup);
if (PIM_TYPE(pim_typever) != 7) { /*not for Graft-ACK*/
2018-01-07 18:47:30 +08:00
ND_PRINT(", holdtime: ");
if (holdtime == 0xffff)
2018-01-07 18:47:30 +08:00
ND_PRINT("infinite");
else
unsigned_relts_print(ndo, holdtime);
}
bp += 4; len -= 4;
for (i = 0; i < ngroup; i++) {
2018-01-07 18:47:30 +08:00
ND_PRINT("\n\t group #%u: ", i+1);
if ((advance = pimv2_addr_print(ndo, bp, len, pimv2_group, pimv2_addr_len, 0)) < 0)
goto trunc;
bp += advance; len -= advance;
if (len < 4)
goto trunc;
ND_TCHECK_4(bp);
njoin = GET_BE_U_2(bp);
nprune = GET_BE_U_2(bp + 2);
2018-01-07 18:47:30 +08:00
ND_PRINT(", joined sources: %u, pruned sources: %u", njoin, nprune);
bp += 4; len -= 4;
for (j = 0; j < njoin; j++) {
2018-01-07 18:47:30 +08:00
ND_PRINT("\n\t joined source #%u: ", j+1);
if ((advance = pimv2_addr_print(ndo, bp, len, pimv2_source, pimv2_addr_len, 0)) < 0)
goto trunc;
bp += advance; len -= advance;
}
for (j = 0; j < nprune; j++) {
2018-01-07 18:47:30 +08:00
ND_PRINT("\n\t pruned source #%u: ", j+1);
if ((advance = pimv2_addr_print(ndo, bp, len, pimv2_source, pimv2_addr_len, 0)) < 0)
goto trunc;
bp += advance; len -= advance;
}
}
break;
}
case PIMV2_TYPE_BOOTSTRAP:
2001-01-28 16:27:28 +08:00
{
u_int i, j, frpcnt;
/* Fragment Tag, Hash Mask len, and BSR-priority */
if (len < 2)
goto trunc;
ND_PRINT(" tag=%x", GET_BE_U_2(bp));
bp += 2;
len -= 2;
if (len < 1)
goto trunc;
ND_PRINT(" hashmlen=%u", GET_U_1(bp));
if (len < 2)
goto trunc;
2017-12-04 01:18:58 +08:00
ND_TCHECK_1(bp + 2);
ND_PRINT(" BSRprio=%u", GET_U_1(bp + 1));
bp += 2;
len -= 2;
/* Encoded-Unicast-BSR-Address */
2018-01-07 18:47:30 +08:00
ND_PRINT(" BSR=");
if ((advance = pimv2_addr_print(ndo, bp, len, pimv2_unicast, pimv2_addr_len, 0)) < 0)
goto trunc;
bp += advance;
len -= advance;
for (i = 0; len > 0; i++) {
/* Encoded-Group Address */
2018-01-07 18:47:30 +08:00
ND_PRINT(" (group%u: ", i);
if ((advance = pimv2_addr_print(ndo, bp, len, pimv2_group, pimv2_addr_len, 0)) < 0)
goto trunc;
bp += advance;
len -= advance;
/* RP-Count, Frag RP-Cnt, and rsvd */
if (len < 1)
goto trunc;
ND_PRINT(" RPcnt=%u", GET_U_1(bp));
if (len < 2)
goto trunc;
frpcnt = GET_U_1(bp + 1);
2018-01-07 18:47:30 +08:00
ND_PRINT(" FRPcnt=%u", frpcnt);
if (len < 4)
goto trunc;
bp += 4;
len -= 4;
for (j = 0; j < frpcnt && len > 0; j++) {
/* each RP info */
2018-01-07 18:47:30 +08:00
ND_PRINT(" RP%u=", j);
if ((advance = pimv2_addr_print(ndo, bp, len,
pimv2_unicast,
pimv2_addr_len,
0)) < 0)
goto trunc;
bp += advance;
len -= advance;
if (len < 2)
goto trunc;
2018-01-07 18:47:30 +08:00
ND_PRINT(",holdtime=");
unsigned_relts_print(ndo,
GET_BE_U_2(bp));
if (len < 3)
goto trunc;
ND_PRINT(",prio=%u", GET_U_1(bp + 2));
if (len < 4)
goto trunc;
bp += 4;
len -= 4;
}
2018-01-07 18:47:30 +08:00
ND_PRINT(")");
}
break;
2001-01-28 16:27:28 +08:00
}
case PIMV2_TYPE_ASSERT:
2018-01-07 18:47:30 +08:00
ND_PRINT(" group=");
if ((advance = pimv2_addr_print(ndo, bp, len, pimv2_group, pimv2_addr_len, 0)) < 0)
goto trunc;
bp += advance; len -= advance;
2018-01-07 18:47:30 +08:00
ND_PRINT(" src=");
if ((advance = pimv2_addr_print(ndo, bp, len, pimv2_unicast, pimv2_addr_len, 0)) < 0)
goto trunc;
bp += advance; len -= advance;
if (len < 8)
goto trunc;
ND_TCHECK_8(bp);
if (GET_U_1(bp) & 0x80)
2018-01-07 18:47:30 +08:00
ND_PRINT(" RPT");
ND_PRINT(" pref=%u", GET_BE_U_4(bp) & 0x7fffffff);
ND_PRINT(" metric=%u", GET_BE_U_4(bp + 4));
break;
case PIMV2_TYPE_CANDIDATE_RP:
2001-01-28 16:27:28 +08:00
{
u_int i, pfxcnt;
/* Prefix-Cnt, Priority, and Holdtime */
if (len < 1)
goto trunc;
ND_PRINT(" prefix-cnt=%u", GET_U_1(bp));
pfxcnt = GET_U_1(bp);
if (len < 2)
goto trunc;
ND_PRINT(" prio=%u", GET_U_1(bp + 1));
if (len < 4)
goto trunc;
2018-01-07 18:47:30 +08:00
ND_PRINT(" holdtime=");
unsigned_relts_print(ndo, GET_BE_U_2(bp + 2));
bp += 4;
len -= 4;
/* Encoded-Unicast-RP-Address */
2018-01-07 18:47:30 +08:00
ND_PRINT(" RP=");
if ((advance = pimv2_addr_print(ndo, bp, len, pimv2_unicast, pimv2_addr_len, 0)) < 0)
goto trunc;
bp += advance;
len -= advance;
/* Encoded-Group Addresses */
for (i = 0; i < pfxcnt && len > 0; i++) {
2018-01-07 18:47:30 +08:00
ND_PRINT(" Group%u=", i);
if ((advance = pimv2_addr_print(ndo, bp, len, pimv2_group, pimv2_addr_len, 0)) < 0)
goto trunc;
bp += advance;
len -= advance;
}
break;
2001-01-28 16:27:28 +08:00
}
case PIMV2_TYPE_PRUNE_REFRESH:
2018-01-07 18:47:30 +08:00
ND_PRINT(" src=");
if ((advance = pimv2_addr_print(ndo, bp, len, pimv2_unicast, pimv2_addr_len, 0)) < 0)
goto trunc;
bp += advance;
len -= advance;
2018-01-07 18:47:30 +08:00
ND_PRINT(" grp=");
if ((advance = pimv2_addr_print(ndo, bp, len, pimv2_group, pimv2_addr_len, 0)) < 0)
goto trunc;
bp += advance;
len -= advance;
2018-01-07 18:47:30 +08:00
ND_PRINT(" forwarder=");
if ((advance = pimv2_addr_print(ndo, bp, len, pimv2_unicast, pimv2_addr_len, 0)) < 0)
goto trunc;
bp += advance;
len -= advance;
if (len < 2)
goto trunc;
2018-01-07 18:47:30 +08:00
ND_PRINT(" TUNR ");
unsigned_relts_print(ndo, GET_BE_U_2(bp));
break;
case PIMV2_TYPE_DF_ELECTION:
subtype = PIM_SUBTYPE(GET_U_1(pim->pim_rsv));
ND_PRINT("\n\t %s,", tok2str( pimv2_df_election_flag_values,
"Unknown", subtype) );
ND_PRINT(" rpa=");
if ((advance = pimv2_addr_print(ndo, bp, len, pimv2_unicast, pimv2_addr_len, 0)) < 0) {
goto trunc;
}
bp += advance;
len -= advance;
ND_PRINT(" sender pref=%u", GET_BE_U_4(bp) );
ND_PRINT(" sender metric=%u", GET_BE_U_4(bp + 4));
bp += 8;
len -= 8;
switch (subtype) {
case PIMV2_DF_ELECTION_BACKOFF:
case PIMV2_DF_ELECTION_PASS:
ND_PRINT("\n\t %s addr=", PIMV2_DF_ELECTION_PASS_BACKOFF_STR(subtype));
if ((advance = pimv2_addr_print(ndo, bp, len, pimv2_unicast, pimv2_addr_len, 0)) < 0) {
goto trunc;
}
bp += advance;
len -= advance;
ND_PRINT(" %s pref=%u", PIMV2_DF_ELECTION_PASS_BACKOFF_STR(subtype), GET_BE_U_4(bp) );
ND_PRINT(" %s metric=%u", PIMV2_DF_ELECTION_PASS_BACKOFF_STR(subtype), GET_BE_U_4(bp + 4));
bp += 8;
len -= 8;
if (subtype == PIMV2_DF_ELECTION_BACKOFF) {
ND_PRINT(" interval %dms", GET_BE_U_2(bp));
}
break;
default:
break;
}
break;
default:
2018-01-07 18:47:30 +08:00
ND_PRINT(" [type %u]", PIM_TYPE(pim_typever));
break;
}
return;
trunc:
nd_print_trunc(ndo);
}