summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorStuart Henderson <sthen@cvs.openbsd.org>2009-09-07 12:25:37 +0000
committerStuart Henderson <sthen@cvs.openbsd.org>2009-09-07 12:25:37 +0000
commit43b9c12d62824e91cc08d2c885aa7ef1a09be84a (patch)
treed0c6197251dd40bd80d80095f0b2ea8aa2893670
parent3f872ca110aa5111df3aa0096bee73d68ea560eb (diff)
Teach the printer about capabilities, new cease codes, end-of-rib marker,
and handle 32-bit ASN. ok claudio@
-rw-r--r--usr.sbin/tcpdump/print-bgp.c378
1 files changed, 324 insertions, 54 deletions
diff --git a/usr.sbin/tcpdump/print-bgp.c b/usr.sbin/tcpdump/print-bgp.c
index 55fdf30cc09..9bfb1789839 100644
--- a/usr.sbin/tcpdump/print-bgp.c
+++ b/usr.sbin/tcpdump/print-bgp.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: print-bgp.c,v 1.11 2009/01/29 09:46:32 bluhm Exp $ */
+/* $OpenBSD: print-bgp.c,v 1.12 2009/09/07 12:25:36 sthen Exp $ */
/*
* Copyright (C) 1999 WIDE Project.
@@ -31,7 +31,7 @@
#ifndef lint
static const char rcsid[] =
- "@(#) $Id: print-bgp.c,v 1.11 2009/01/29 09:46:32 bluhm Exp $";
+ "@(#) $Id: print-bgp.c,v 1.12 2009/09/07 12:25:36 sthen Exp $";
#endif
#include <sys/param.h>
@@ -62,6 +62,7 @@ struct bgp {
#define BGP_UPDATE 2
#define BGP_NOTIFICATION 3
#define BGP_KEEPALIVE 4
+#define BGP_ROUTE_REFRESH 5
struct bgp_open {
u_int8_t bgpo_marker[16];
@@ -81,8 +82,11 @@ struct bgp_opt {
u_int8_t bgpopt_len;
/* variable length */
};
+#define BGP_OPT_CAP 2
#define BGP_OPT_SIZE 2 /* some compilers may pad to 4 bytes */
+#define BGP_UPDATE_MINSIZE 23
+
struct bgp_notification {
u_int8_t bgpn_marker[16];
u_int16_t bgpn_len;
@@ -93,6 +97,16 @@ struct bgp_notification {
};
#define BGP_NOTIFICATION_SIZE 21 /* unaligned */
+struct bgp_route_refresh {
+ u_int8_t bgp_marker[16];
+ u_int16_t len;
+ u_int8_t type;
+ u_int8_t afi[2]; /* unaligned; should be u_int16_t */
+ u_int8_t res;
+ u_int8_t safi;
+};
+#define BGP_ROUTE_REFRESH_SIZE 23
+
struct bgp_attr {
u_int8_t bgpa_flags;
u_int8_t bgpa_type;
@@ -122,11 +136,35 @@ struct bgp_attr {
#define BGPTYPE_RCID_PATH 13 /* RFC1863 */
#define BGPTYPE_MP_REACH_NLRI 14 /* RFC2283 */
#define BGPTYPE_MP_UNREACH_NLRI 15 /* RFC2283 */
+#define BGPTYPE_EXTD_COMMUNITIES 16 /* RFC4360 */
+#define BGPTYPE_AS4_PATH 17 /* RFC4893 */
+#define BGPTYPE_AGGREGATOR4 18 /* RFC4893 */
+
+#define BGP_AS_SET 1
+#define BGP_AS_SEQUENCE 2
+#define BGP_CONFED_AS_SEQUENCE 3 /* draft-ietf-idr-rfc3065bis-01 */
+#define BGP_CONFED_AS_SET 4 /* draft-ietf-idr-rfc3065bis-01 */
+
+static struct tok bgp_as_path_segment_open_values[] = {
+ { BGP_AS_SET, " {" },
+ { BGP_AS_SEQUENCE, " " },
+ { BGP_CONFED_AS_SEQUENCE, " (" },
+ { BGP_CONFED_AS_SET, " ({" },
+ { 0, NULL},
+};
+
+static struct tok bgp_as_path_segment_close_values[] = {
+ { BGP_AS_SET, "}" },
+ { BGP_AS_SEQUENCE, "" },
+ { BGP_CONFED_AS_SEQUENCE, ")" },
+ { BGP_CONFED_AS_SET, "})" },
+ { 0, NULL},
+};
#define BGP_MP_NLRI_MINSIZE 3
static const char *bgptype[] = {
- NULL, "OPEN", "UPDATE", "NOTIFICATION", "KEEPALIVE",
+ NULL, "OPEN", "UPDATE", "NOTIFICATION", "KEEPALIVE", "ROUTE-REFRESH",
};
#define bgp_type(x) num_or_str(bgptype, sizeof(bgptype)/sizeof(bgptype[0]), (x))
@@ -136,29 +174,53 @@ static const char *bgpopt_type[] = {
#define bgp_opttype(x) \
num_or_str(bgpopt_type, sizeof(bgpopt_type)/sizeof(bgpopt_type[0]), (x))
+#define BGP_CAPCODE_MP 1
+#define BGP_CAPCODE_RESTART 64 /* draft-ietf-idr-restart-05 */
+#define BGP_CAPCODE_AS4 65 /* RFC4893 */
+
+static const char *bgp_capcode[] = {
+ NULL, "MULTI_PROTOCOL", "ROUTE_REFRESH",
+ /* 3: RFC5291 */ "OUTBOUND_ROUTE_FILTERING",
+ /* 4: RFC3107 */ "MULTIPLE_ROUTES",
+ /* 5: RFC5549 */ "EXTENDED_NEXTHOP_ENCODING",
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+ /* 64: RFC4724 */ "GRACEFUL_RESTART",
+ /* 65: RFC4893 */ "AS4", 0,
+ /* 67: [Chen] */ "DYNAMIC_CAPABILITY",
+ /* 68: [Appanna] */ "MULTISESSION",
+ /* 69: [draft-ietf-idr-add-paths] */ "ADD-PATH",
+};
+
+#define bgp_capcode(x) \
+ num_or_str(bgp_capcode, sizeof(bgp_capcode)/sizeof(bgp_capcode[0]), (x))
+
+#define BGP_NOTIFY_MAJOR_CEASE 6
static const char *bgpnotify_major[] = {
NULL, "Message Header Error",
"OPEN Message Error", "UPDATE Message Error",
"Hold Timer Expired", "Finite State Machine Error",
- "Cease",
+ "Cease", "Capability Message Error",
};
#define bgp_notify_major(x) \
num_or_str(bgpnotify_major, \
sizeof(bgpnotify_major)/sizeof(bgpnotify_major[0]), (x))
-static const char *bgpnotify_minor_1[] = {
+static const char *bgpnotify_minor_msg[] = {
NULL, "Connection Not Synchronized",
"Bad Message Length", "Bad Message Type",
};
-static const char *bgpnotify_minor_2[] = {
+static const char *bgpnotify_minor_open[] = {
NULL, "Unsupported Version Number",
"Bad Peer AS", "Bad BGP Identifier",
"Unsupported Optional Parameter", "Authentication Failure",
- "Unacceptable Hold Time",
+ "Unacceptable Hold Time", "Unsupported Capability",
};
-static const char *bgpnotify_minor_3[] = {
+static const char *bgpnotify_minor_update[] = {
NULL, "Malformed Attribute List",
"Unrecognized Well-known Attribute", "Missing Well-known Attribute",
"Attribute Flags Error", "Attribute Length Error",
@@ -167,13 +229,32 @@ static const char *bgpnotify_minor_3[] = {
"Invalid Network Field", "Malformed AS_PATH",
};
+/* RFC 4486 */
+#define BGP_NOTIFY_MINOR_CEASE_MAXPRFX 1
+static const char *bgpnotify_minor_cease[] = {
+ NULL, "Maximum Number of Prefixes Reached", "Administratively Shutdown",
+ "Peer De-configured", "Administratively Reset", "Connection Rejected",
+ "Other Configuration Change", "Connection Collision Resolution",
+ "Out of Resources",
+};
+
+static const char *bgpnotify_minor_cap[] = {
+ NULL, "Invalid Action Value", "Invalid Capability Length",
+ "Malformed Capability Value", "Unsupported Capability Code",
+};
+
static const char **bgpnotify_minor[] = {
- NULL, bgpnotify_minor_1, bgpnotify_minor_2, bgpnotify_minor_3,
+ NULL, bgpnotify_minor_msg, bgpnotify_minor_open, bgpnotify_minor_update,
};
static const int bgpnotify_minor_siz[] = {
- 0, sizeof(bgpnotify_minor_1)/sizeof(bgpnotify_minor_1[0]),
- sizeof(bgpnotify_minor_2)/sizeof(bgpnotify_minor_2[0]),
- sizeof(bgpnotify_minor_3)/sizeof(bgpnotify_minor_3[0]),
+ 0,
+ sizeof(bgpnotify_minor_msg)/sizeof(bgpnotify_minor_msg[0]),
+ sizeof(bgpnotify_minor_open)/sizeof(bgpnotify_minor_open[0]),
+ sizeof(bgpnotify_minor_update)/sizeof(bgpnotify_minor_update[0]),
+ 0,
+ 0,
+ sizeof(bgpnotify_minor_cease)/sizeof(bgpnotify_minor_cease[0]),
+ sizeof(bgpnotify_minor_cap)/sizeof(bgpnotify_minor_cap[0]),
};
static const char *bgpattr_origin[] = {
@@ -188,6 +269,7 @@ static const char *bgpattr_type[] = {
"MULTI_EXIT_DISC", "LOCAL_PREF", "ATOMIC_AGGREGATE", "AGGREGATOR",
"COMMUNITIES", "ORIGINATOR_ID", "CLUSTER_LIST", "DPA",
"ADVERTISERS", "RCID_PATH", "MP_REACH_NLRI", "MP_UNREACH_NLRI",
+ "EXTD_COMMUNITIES", "AS4_PATH", "AGGREGATOR4",
};
#define bgp_attr_type(x) \
num_or_str(bgpattr_type, \
@@ -196,6 +278,13 @@ static const char *bgpattr_type[] = {
/* Subsequent address family identifier, RFC2283 section 7 */
static const char *bgpattr_nlri_safi[] = {
"Reserved", "Unicast", "Multicast", "Unicast+Multicast",
+ "labeled Unicast", /* MPLS BGP RFC3107 */
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+ /* 64-66: MPLS BGP RFC3107 */
+ "Tunnel", "VPLS", "MDT",
};
#define bgp_attr_nlri_safi(x) \
num_or_str(bgpattr_nlri_safi, \
@@ -263,6 +352,7 @@ decode_prefix4(const u_char *pd, char *buf, u_int buflen)
* prefix length is in bits; packet only contains
* enough bytes of address to contain this many bits
*/
+ plen = pd[0];
if (plen < 0 || 32 < plen)
return -1;
memset(&addr, 0, sizeof(addr));
@@ -321,11 +411,13 @@ bgp_attr_print(const struct bgp_attr *attr, const u_char *dat, int len)
u_int16_t af;
u_int8_t safi, snpa;
int advance;
- int tlen;
+ int tlen, asn_bytes;
const u_char *p;
char buf[MAXHOSTNAMELEN + 100];
p = dat;
+ tlen = len;
+ asn_bytes = 0;
switch (attr->bgpa_type) {
case BGPTYPE_ORIGIN:
@@ -336,7 +428,18 @@ bgp_attr_print(const struct bgp_attr *attr, const u_char *dat, int len)
printf(" %s", bgp_attr_origin(p[0]));
}
break;
+ case BGPTYPE_AS4_PATH:
+ asn_bytes = 4;
+ /* FALLTHROUGH */
case BGPTYPE_AS_PATH:
+ /*
+ * 2-byte speakers will receive AS4_PATH as well AS_PATH (2-byte).
+ * 4-byte speakers will only receive AS_PATH but it will be 4-byte.
+ * To identify which is the case, compare the length of the path
+ * segment value in bytes, with the path segment length from the
+ * message (counted in # of AS)
+ */
+
if (len % 2) {
printf(" invalid len");
break;
@@ -346,28 +449,31 @@ bgp_attr_print(const struct bgp_attr *attr, const u_char *dat, int len)
break;
}
while (p < dat + len) {
- /*
- * under RFC1965, p[0] means:
- * 1: AS_SET 2: AS_SEQUENCE
- * 3: AS_CONFED_SET 4: AS_CONFED_SEQUENCE
- */
- printf(" ");
- TCHECK(p[1]);
- if (p[0] == 3 || p[0] == 4)
- printf("confed");
- printf("%s", (p[0] & 1) ? "{" : "");
- for (i = 0; i < p[1] * 2; i += 2) {
- TCHECK2(p[2 + i], 2); /* ASN is 2-bytes */
- printf("%s%u", i == 0 ? "" : " ",
- EXTRACT_16BITS(&p[2 + i]));
+ TCHECK(p[0]);
+ if (asn_bytes == 0) {
+ asn_bytes = (len-2)/p[1];
}
- printf("%s", (p[0] & 1) ? "}" : "");
- /*
- * 1 byte for attr type, 1 byte for path length,
- * plus size of ASN (2-bytes for now) * no of ASN
- *
- */
- p += 2 + p[1] * 2;
+ printf("%s",
+ tok2str(bgp_as_path_segment_open_values,
+ "?", p[0]));
+
+ for (i = 0; i < p[1] * asn_bytes; i += asn_bytes) {
+ TCHECK2(p[2 + i], asn_bytes);
+ printf("%s", i == 0 ? "" : " ");
+ if (asn_bytes == 2 || EXTRACT_16BITS(&p[2 + i]))
+ printf("%u%s",
+ EXTRACT_16BITS(&p[2 + i]),
+ asn_bytes == 4 ? "." : "");
+ if (asn_bytes == 4)
+ printf("%u",
+ EXTRACT_16BITS(&p[2 + i + 2]));
+ }
+ TCHECK(p[0]);
+ printf("%s",
+ tok2str(bgp_as_path_segment_close_values,
+ "?", p[0]));
+ TCHECK(p[1]);
+ p += 2 + p[1] * asn_bytes;
}
break;
case BGPTYPE_NEXT_HOP:
@@ -391,24 +497,34 @@ bgp_attr_print(const struct bgp_attr *attr, const u_char *dat, int len)
if (len != 0)
printf(" invalid len");
break;
+ case BGPTYPE_AGGREGATOR4:
case BGPTYPE_AGGREGATOR:
- if (len != 6) {
+ /*
+ * like AS_PATH/AS4_PATH, AGGREGATOR can contain
+ * either 2-byte or 4-byte ASN, and AGGREGATOR4
+ * always contains 4-byte ASN.
+ */
+ if (len != 6 && len != 8) {
printf(" invalid len");
break;
}
- TCHECK2(p[0], 6);
- printf(" AS #%u, origin %s", EXTRACT_16BITS(p),
- getname(p + 2));
+ TCHECK2(p[0], len);
+ printf(" AS #");
+ if (len == 6 || EXTRACT_16BITS(p))
+ printf("%u%s", EXTRACT_16BITS(p), len == 8 ? "." : "");
+ if (len == 8)
+ printf("%u", EXTRACT_16BITS(p+2));
+ printf(", origin %s", getname(p+len-4));
break;
case BGPTYPE_COMMUNITIES:
if (len % 4) {
printf(" invalid len");
break;
}
- for (i = 0; i < len; i += 4) {
+ while (tlen>0) {
u_int32_t comm;
- TCHECK2(p[i], 4);
- comm = EXTRACT_32BITS(&p[i]);
+ TCHECK2(p[0], 4);
+ comm = EXTRACT_32BITS(p);
switch (comm) {
case BGP_COMMUNITY_NO_EXPORT:
printf(" NO_EXPORT");
@@ -427,6 +543,30 @@ bgp_attr_print(const struct bgp_attr *attr, const u_char *dat, int len)
(comm >> 16) & 0xffff, comm & 0xffff);
break;
}
+ tlen -= 4;
+ p += 4;
+ }
+ break;
+ case BGPTYPE_ORIGINATOR_ID:
+ if (len != 4) {
+ printf(" invalid len");
+ break;
+ }
+ TCHECK2(p[0], 4);
+ printf("%s",getname(p));
+ break;
+ case BGPTYPE_CLUSTER_LIST:
+ if (len % 4) {
+ printf(" invalid len");
+ break;
+ }
+ while (tlen>0) {
+ TCHECK2(p[0], 4);
+ printf(" %s%s",
+ getname(p),
+ (tlen>4) ? ", " : "");
+ tlen -=4;
+ p +=4;
}
break;
case BGPTYPE_MP_REACH_NLRI:
@@ -434,7 +574,7 @@ bgp_attr_print(const struct bgp_attr *attr, const u_char *dat, int len)
af = EXTRACT_16BITS(p);
safi = p[2];
if (safi >= 128)
- printf(" %s vendor specific,", af_name(af));
+ printf(" %s vendor specific %u,", af_name(af), safi);
else {
printf(" %s %s,", af_name(af),
bgp_attr_nlri_safi(safi));
@@ -524,7 +664,7 @@ bgp_attr_print(const struct bgp_attr *attr, const u_char *dat, int len)
af = EXTRACT_16BITS(p);
safi = p[2];
if (safi >= 128)
- printf(" %s vendor specific,", af_name(af));
+ printf(" %s vendor specific %u,", af_name(af), safi);
else {
printf(" %s %s,", af_name(af),
bgp_attr_nlri_safi(safi));
@@ -571,7 +711,7 @@ bgp_open_print(const u_char *dat, int length)
struct bgp_open bgpo;
struct bgp_opt bgpopt;
const u_char *opt;
- int i;
+ int i,cap_type,cap_len,tcap_len,cap_offset;
TCHECK2(dat[0], BGP_OPEN_SIZE);
memcpy(&bgpo, dat, BGP_OPEN_SIZE);
@@ -582,6 +722,10 @@ bgp_open_print(const u_char *dat, int length)
printf(" ID %s,", getname((u_char *)&bgpo.bgpo_id));
printf(" Option length %u", bgpo.bgpo_optlen);
+ /* sanity checking */
+ if ((length < bgpo.bgpo_optlen+BGP_OPEN_SIZE) || (!bgpo.bgpo_optlen))
+ return;
+
/* ugly! */
opt = &((const struct bgp_open *)dat)->bgpo_optlen;
opt++;
@@ -595,10 +739,71 @@ bgp_open_print(const u_char *dat, int length)
break;
}
- printf(" (option %s, len=%d)", bgp_opttype(bgpopt.bgpopt_type),
- bgpopt.bgpopt_len);
+ if (i == 0)
+ printf(" ("); /* ) */
+ else
+ printf(" ");
+
+ switch(bgpopt.bgpopt_type) {
+ case BGP_OPT_CAP:
+ cap_type=opt[i+BGP_OPT_SIZE];
+ cap_len=opt[i+BGP_OPT_SIZE+1];
+ tcap_len=cap_len;
+ printf("(CAP %s", /* ) */
+ bgp_capcode(cap_type));
+
+ switch(cap_type) {
+ case BGP_CAPCODE_MP:
+ printf(" [%s %s]",
+ af_name(EXTRACT_16BITS(opt+i+BGP_OPT_SIZE+2)),
+ bgp_attr_nlri_safi(opt[i+BGP_OPT_SIZE+5]));
+ break;
+ case BGP_CAPCODE_RESTART:
+ printf(" [%s], Time %us",
+ ((opt[i+BGP_OPT_SIZE+2])&0x80) ? "R" : "none",
+ EXTRACT_16BITS(opt+i+BGP_OPT_SIZE+2)&0xfff);
+ tcap_len-=2;
+ cap_offset=4;
+ while(tcap_len>=4) {
+ printf(" (%s %s)%s",
+ af_name(EXTRACT_16BITS(
+ opt+i+BGP_OPT_SIZE+cap_offset)),
+ bgp_attr_nlri_safi(
+ opt[i+BGP_OPT_SIZE+cap_offset+2]),
+ ((opt[i+BGP_OPT_SIZE+cap_offset+3])&0x80) ?
+ " forwarding state preserved" : "" );
+ tcap_len-=4;
+ cap_offset+=4;
+ }
+ break;
+ case BGP_CAPCODE_AS4:
+ printf(" #");
+ if (EXTRACT_16BITS(opt+i+BGP_OPT_SIZE+2))
+ printf("%u.",
+ EXTRACT_16BITS(opt+i+BGP_OPT_SIZE+2));
+ printf("%u",
+ EXTRACT_16BITS(opt+i+BGP_OPT_SIZE+4));
+ break;
+ default:
+ break;
+ }
+ if (vflag > 1) {
+ TCHECK2(opt[i+BGP_OPT_SIZE+2],cap_len);
+ }
+ /* ( */
+ printf(")");
+ break;
+ default:
+ printf(" (option %s, len=%u)",
+ bgp_opttype(bgpopt.bgpopt_type),
+ bgpopt.bgpopt_len);
+ break;
+ }
+
i += BGP_OPT_SIZE + bgpopt.bgpopt_len;
}
+ /* ( */
+ printf(")");
return;
trunc:
printf("[|BGP]");
@@ -640,12 +845,11 @@ bgp_update_print(const u_char *dat, int length)
while(i < 2 + len) {
wpfx = decode_prefix4(&p[i], buf, sizeof(buf));
- if (wpfx == -2)
- goto trunc;
- else if (wpfx < 0) {
+ if (wpfx = -1) {
printf(" (illegal prefix length)");
break;
- }
+ } else if (wpfx == -2)
+ goto trunc;
i += wpfx;
printf(" %s", buf);
}
@@ -657,6 +861,11 @@ bgp_update_print(const u_char *dat, int length)
TCHECK2(p[0], 2);
len = EXTRACT_16BITS(p);
+ if (len == 0 && length == BGP_UPDATE_MINSIZE) {
+ printf(" End-of-Rib Marker (empty NLRI)");
+ return;
+ }
+
if (len) {
/* do something more useful!*/
i = 2;
@@ -709,12 +918,11 @@ bgp_update_print(const u_char *dat, int length)
while (dat + length > p) {
char buf[MAXHOSTNAMELEN + 100];
i = decode_prefix4(p, buf, sizeof(buf));
- if (i == -2)
- goto trunc;
- else if (i < 0) {
+ if (i == -1) {
printf(" (illegal prefix length)");
break;
- }
+ } else if (i == -2)
+ goto trunc;
printf(" %s", buf);
p += i;
}
@@ -731,13 +939,63 @@ static void
bgp_notification_print(const u_char *dat, int length)
{
struct bgp_notification bgpn;
+ u_int16_t af;
+ u_int8_t safi;
+ const u_char *p;
TCHECK2(dat[0], BGP_NOTIFICATION_SIZE);
memcpy(&bgpn, dat, BGP_NOTIFICATION_SIZE);
+ /* sanity checking */
+ if (length<BGP_NOTIFICATION_SIZE)
+ return;
+
printf(": error %s,", bgp_notify_major(bgpn.bgpn_major));
printf(" subcode %s",
bgp_notify_minor(bgpn.bgpn_major, bgpn.bgpn_minor));
+
+ if (bgpn.bgpn_major == BGP_NOTIFY_MAJOR_CEASE) {
+ /*
+ * RFC 4486: optional maxprefix subtype of 7 bytes
+ * may contain AFI, SAFI and MAXPREFIXES
+ */
+ if(bgpn.bgpn_minor == BGP_NOTIFY_MINOR_CEASE_MAXPRFX &&
+ length >= BGP_NOTIFICATION_SIZE + 7) {
+
+ p = dat + BGP_NOTIFICATION_SIZE;
+ TCHECK2(*p, 7);
+
+ af = EXTRACT_16BITS(p);
+ safi = p[2];
+ printf(" %s %s,", af_name(af),
+ bgp_attr_nlri_safi(safi));
+
+ printf(" Max Prefixes: %u", EXTRACT_32BITS(p+3));
+ }
+ }
+
+ return;
+trunc:
+ printf("[|BGP]");
+}
+
+static void
+bgp_route_refresh_print(const u_char *dat, int length)
+{
+ const struct bgp_route_refresh *bgp_route_refresh_header;
+
+ TCHECK2(dat[0], BGP_ROUTE_REFRESH_SIZE);
+
+ /* sanity checking */
+ if (length<BGP_ROUTE_REFRESH_SIZE)
+ return;
+
+ bgp_route_refresh_header = (const struct bgp_route_refresh *)dat;
+
+ printf(" (%s %s)",
+ af_name(EXTRACT_16BITS(&bgp_route_refresh_header->afi)),
+ bgp_attr_nlri_safi(bgp_route_refresh_header->safi));
+
return;
trunc:
printf("[|BGP]");
@@ -762,6 +1020,13 @@ bgp_header_print(const u_char *dat, int length)
case BGP_NOTIFICATION:
bgp_notification_print(dat, length);
break;
+ case BGP_KEEPALIVE:
+ break;
+ case BGP_ROUTE_REFRESH:
+ bgp_route_refresh_print(dat, length);
+ default:
+ TCHECK2(*dat, length);
+ break;
}
/* ( */
@@ -822,6 +1087,11 @@ bgp_print(const u_char *dat, int length)
printf("\n\t");
else
printf(" ");
+ if (hlen < BGP_SIZE) {
+ printf("\n[|BGP Bogus header length %u < %u]",
+ hlen, BGP_SIZE);
+ break;
+ }
if (TTEST2(p[0], hlen)) {
if (!bgp_header_print(p, hlen))
return;