forked from osmocom/wireshark
cc36f0b931
the following: It is now possible to enable/disable a particular protocol decoding (i.e. the protocol dissector is void or not). When a protocol is disabled, it is displayed as Data and of course, all linked sub-protocols are disabled as well. Disabling a protocol could be interesting: - in case of buggy dissectors - in case of wrong heuristics - for performance reasons - to decode the data as another protocol (TODO) Currently (if I am not wrong), all dissectors but NFS can be disabled (and dissectors that do not register protocols :-) I do not like the way the RPC sub-dissectors are disabled (in the sub-dissectors) since this could be done in the RPC dissector itself, knowing the sub-protocol hfinfo entry (this is why, I've not modified the NFS one yet). Two functions are added in proto.c : gboolean proto_is_protocol_enabled(int n); void proto_set_decoding(int n, gboolean enabled); and two MACROs which can be used in dissectors: OLD_CHECK_DISPLAY_AS_DATA(index, pd, offset, fd, tree) CHECK_DISPLAY_AS_DATA(index, tvb, pinfo, tree) See also the XXX in proto_dlg.c and proto.c around the new functions. svn path=/trunk/; revision=2267
611 lines
15 KiB
C
611 lines
15 KiB
C
/* packet-pim.c
|
|
* Routines for PIM disassembly
|
|
* (c) Copyright Jun-ichiro itojun Hagino <itojun@itojun.org>
|
|
*
|
|
* $Id: packet-pim.c,v 1.17 2000/08/13 14:08:37 deniel Exp $
|
|
*
|
|
* Ethereal - Network traffic analyzer
|
|
* By Gerald Combs <gerald@zing.org>
|
|
* Copyright 1998 Gerald Combs
|
|
*
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#ifdef HAVE_SYS_TYPES_H
|
|
#include <sys/types.h>
|
|
#endif
|
|
|
|
#ifdef HAVE_NETINET_IN_H
|
|
#include <netinet/in.h>
|
|
#endif
|
|
|
|
#include <glib.h>
|
|
|
|
#ifdef NEED_SNPRINTF_H
|
|
# include "snprintf.h"
|
|
#endif
|
|
|
|
#include "packet.h"
|
|
#include "packet-ip.h"
|
|
#include "packet-ipv6.h"
|
|
|
|
#ifndef offsetof
|
|
#define offsetof(type, member) ((size_t)(&((type *)0)->member))
|
|
#endif
|
|
|
|
struct pim {
|
|
guint8 pim_typever;
|
|
#define PIM_TYPE(x) ((x) & 0x0f)
|
|
#define PIM_VER(x) (((x) & 0xf0) >> 4)
|
|
guint8 pim_rsv; /* Reserved */
|
|
guint16 pim_cksum; /* IP style check sum */
|
|
};
|
|
|
|
enum pimv2_addrtype {
|
|
pimv2_unicast, pimv2_group, pimv2_source
|
|
};
|
|
|
|
static int proto_pim = -1;
|
|
static int hf_pim_version = -1;
|
|
static int hf_pim_type = -1;
|
|
static int hf_pim_cksum = -1;
|
|
|
|
static gint ett_pim = -1;
|
|
|
|
static const char *
|
|
dissect_pim_addr(const u_char *bp, const u_char *ep, enum pimv2_addrtype at,
|
|
int *advance) {
|
|
static char buf[512];
|
|
int af;
|
|
int len = 0;
|
|
|
|
if (bp >= ep)
|
|
return NULL;
|
|
|
|
switch (bp[0]) {
|
|
case 1:
|
|
af = 4;
|
|
break;
|
|
case 2:
|
|
af = 6;
|
|
break;
|
|
default:
|
|
return NULL;
|
|
}
|
|
|
|
if (bp[1] != 0)
|
|
return NULL;
|
|
|
|
switch (at) {
|
|
case pimv2_unicast:
|
|
if (af == 4) {
|
|
len = 4;
|
|
if (bp + 2 + len > ep)
|
|
return NULL;
|
|
(void)snprintf(buf, sizeof(buf), "%s", ip_to_str(bp + 2));
|
|
}
|
|
else if (af == 6) {
|
|
len = 16;
|
|
if (bp + 2 + len > ep)
|
|
return NULL;
|
|
(void)snprintf(buf, sizeof(buf), "%s",
|
|
ip6_to_str((struct e_in6_addr *)(bp + 2)));
|
|
}
|
|
if (advance)
|
|
*advance = 2 + len;
|
|
break;
|
|
|
|
case pimv2_group:
|
|
if (af == 4) {
|
|
len = 4;
|
|
if (bp + 4 + len > ep)
|
|
return NULL;
|
|
(void)snprintf(buf, sizeof(buf), "%s/%u", ip_to_str(bp + 4), bp[3]);
|
|
}
|
|
else if (af == 6) {
|
|
len = 16;
|
|
if (bp + 4 + len > ep)
|
|
return NULL;
|
|
(void)snprintf(buf, sizeof(buf), "%s/%u",
|
|
ip6_to_str((struct e_in6_addr *)(bp + 4)), bp[3]);
|
|
}
|
|
if (advance)
|
|
*advance = 4 + len;
|
|
break;
|
|
case pimv2_source:
|
|
if (af == 4) {
|
|
len = 4;
|
|
if (bp + 4 + len > ep)
|
|
return NULL;
|
|
(void)snprintf(buf, sizeof(buf), "%s/%u", ip_to_str(bp + 4), bp[3]);
|
|
}
|
|
else if (af == 6) {
|
|
len = 16;
|
|
if (bp + 4 + len > ep)
|
|
return NULL;
|
|
(void)snprintf(buf, sizeof(buf), "%s/%u",
|
|
ip6_to_str((struct e_in6_addr *)(bp + 4)), bp[3]);
|
|
}
|
|
if (bp[2]) {
|
|
(void)snprintf(buf + strlen(buf), sizeof(buf) - strlen(buf),
|
|
" (%s%s%s)",
|
|
bp[2] & 0x04 ? "S" : "",
|
|
bp[2] & 0x02 ? "W" : "",
|
|
bp[2] & 0x01 ? "R" : "");
|
|
}
|
|
if (advance)
|
|
*advance = 4 + len;
|
|
break;
|
|
default:
|
|
return NULL;
|
|
}
|
|
|
|
return buf;
|
|
}
|
|
|
|
static void
|
|
dissect_pim(const u_char *pd, int offset, frame_data *fd, proto_tree *tree) {
|
|
struct pim pim;
|
|
static const value_string type1vals[] = {
|
|
{ 0, "Query" },
|
|
{ 1, "Register" },
|
|
{ 2, "Register-stop" },
|
|
{ 3, "Join/Prune" },
|
|
{ 4, "RP-Reachable" },
|
|
{ 5, "Assert" },
|
|
{ 6, "Graft" },
|
|
{ 7, "Graft-Ack" },
|
|
{ 8, "Mode" },
|
|
{ 0, NULL },
|
|
};
|
|
static const value_string type2vals[] = {
|
|
{ 0, "Hello" },
|
|
{ 1, "Register" },
|
|
{ 2, "Register-stop" },
|
|
{ 3, "Join/Prune" },
|
|
{ 4, "Bootstrap" },
|
|
{ 5, "Assert" },
|
|
{ 6, "Graft" },
|
|
{ 7, "Graft-Ack" },
|
|
{ 8, "Candidate-RP-Advertisement" },
|
|
{ 0, NULL },
|
|
};
|
|
char *typestr;
|
|
proto_tree *pim_tree = NULL;
|
|
proto_item *ti;
|
|
proto_tree *pimopt_tree = NULL;
|
|
proto_item *tiopt;
|
|
|
|
OLD_CHECK_DISPLAY_AS_DATA(proto_pim, pd, offset, fd, tree);
|
|
|
|
/* avoid alignment problem */
|
|
memcpy(&pim, &pd[offset], sizeof(pim));
|
|
|
|
switch (PIM_VER(pim.pim_typever)) {
|
|
case 1:
|
|
typestr = val_to_str(PIM_TYPE(pim.pim_typever), type1vals, "Unknown");
|
|
break;
|
|
case 2:
|
|
typestr = val_to_str(PIM_TYPE(pim.pim_typever), type2vals, "Unknown");
|
|
break;
|
|
default:
|
|
typestr = "Unknown";
|
|
break;
|
|
}
|
|
|
|
if (check_col(fd, COL_PROTOCOL)) {
|
|
col_add_fstr(fd, COL_PROTOCOL, "PIM version %d",
|
|
PIM_VER(pim.pim_typever));
|
|
}
|
|
if (check_col(fd, COL_INFO))
|
|
col_add_fstr(fd, COL_INFO, "%s", typestr);
|
|
|
|
if (tree) {
|
|
ti = proto_tree_add_item(tree, proto_pim, NullTVB, offset, END_OF_FRAME, FALSE);
|
|
pim_tree = proto_item_add_subtree(ti, ett_pim);
|
|
|
|
proto_tree_add_uint(pim_tree, hf_pim_version, NullTVB, offset, 1,
|
|
PIM_VER(pim.pim_typever));
|
|
proto_tree_add_uint_format(pim_tree, hf_pim_type, NullTVB, offset, 1,
|
|
PIM_TYPE(pim.pim_typever),
|
|
"Type: %s (%u)", typestr, PIM_TYPE(pim.pim_typever));
|
|
|
|
proto_tree_add_uint(pim_tree, hf_pim_cksum, NullTVB,
|
|
offset + offsetof(struct pim, pim_cksum), sizeof(pim.pim_cksum),
|
|
ntohs(pim.pim_cksum));
|
|
|
|
if (sizeof(struct pim) < END_OF_FRAME) {
|
|
tiopt = proto_tree_add_text(pim_tree, NullTVB,
|
|
offset + sizeof(struct pim), END_OF_FRAME,
|
|
"PIM parameters");
|
|
pimopt_tree = proto_item_add_subtree(tiopt, ett_pim);
|
|
} else
|
|
goto done;
|
|
|
|
if (PIM_VER(pim.pim_typever) != 2)
|
|
goto done;
|
|
|
|
/* version 2 decoder */
|
|
switch (PIM_TYPE(pim.pim_typever)) {
|
|
case 0: /*hello*/
|
|
{
|
|
guint16 *w;
|
|
w = (guint16 *)&pd[offset + sizeof(struct pim)];
|
|
while ((guint8 *)w < &pd[offset + END_OF_FRAME]) {
|
|
if (pntohs(&w[0]) == 1 && pntohs(&w[1]) == 2
|
|
&& (guint8 *)&w[3] <= &pd[offset + END_OF_FRAME]) {
|
|
proto_tree_add_text(pimopt_tree, NullTVB, (guint8 *)w - pd, 6,
|
|
"Holdtime: %u%s", pntohs(&w[2]),
|
|
pntohs(&w[2]) == 0xffff ? " (infty)" : "");
|
|
w += 3;
|
|
} else
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 1: /* register */
|
|
{
|
|
const guint8 *ip;
|
|
proto_tree *flag_tree = NULL;
|
|
proto_item *tiflag;
|
|
int flagoff;
|
|
|
|
flagoff = offset + sizeof(struct pim);
|
|
tiflag = proto_tree_add_text(pimopt_tree, NullTVB, flagoff, 4,
|
|
"Flags: 0x%08x", pntohl(&pd[flagoff]));
|
|
flag_tree = proto_item_add_subtree(tiflag, ett_pim);
|
|
proto_tree_add_text(flag_tree, NullTVB, flagoff, 1, "%s",
|
|
decode_boolean_bitfield(pd[flagoff], 0x80000000, 32,
|
|
"Border", "Not border"));
|
|
proto_tree_add_text(flag_tree, NullTVB, flagoff, 1, "%s",
|
|
decode_boolean_bitfield(pd[flagoff], 0x40000000, 32,
|
|
"Null-Register", "Not Null-Register"));
|
|
|
|
ip = &pd[flagoff + sizeof(guint32)];
|
|
switch((*ip & 0xf0) >> 4) {
|
|
case 4: /* IPv4 */
|
|
#if 0
|
|
dissect_ip(pd, ip - pd, fd, tree);
|
|
#else
|
|
dissect_ip(pd, ip - pd, fd, pimopt_tree);
|
|
#endif
|
|
break;
|
|
case 6: /* IPv6 */
|
|
#if 0
|
|
dissect_ipv6(pd, ip - pd, fd, tree);
|
|
#else
|
|
dissect_ipv6(pd, ip - pd, fd, pimopt_tree);
|
|
#endif
|
|
break;
|
|
default:
|
|
proto_tree_add_text(pimopt_tree, NullTVB,
|
|
ip - pd, END_OF_FRAME,
|
|
"Unknown IP version %d", (*ip & 0xf0) >> 4);
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 2: /* register-stop */
|
|
{
|
|
int advance;
|
|
const guint8 *ep;
|
|
const char *s;
|
|
|
|
ep = &pd[offset + END_OF_FRAME];
|
|
offset += sizeof(struct pim);
|
|
s = dissect_pim_addr(&pd[offset], ep, pimv2_group, &advance);
|
|
if (s == NULL)
|
|
break;
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset, advance, "Group: %s", s);
|
|
offset += advance;
|
|
s = dissect_pim_addr(&pd[offset], ep, pimv2_unicast, &advance);
|
|
if (s == NULL)
|
|
break;
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset, advance, "Source: %s", s);
|
|
break;
|
|
}
|
|
|
|
case 3: /* join/prune */
|
|
case 6: /* graft */
|
|
case 7: /* graft-ack */
|
|
{
|
|
int advance;
|
|
const guint8 *ep;
|
|
int off;
|
|
const char *s;
|
|
int ngroup, i, njoin, nprune, j;
|
|
proto_tree *grouptree = NULL;
|
|
proto_item *tigroup;
|
|
proto_tree *subtree = NULL;
|
|
proto_item *tisub;
|
|
|
|
ep = &pd[offset + END_OF_FRAME];
|
|
offset += sizeof(struct pim);
|
|
if (PIM_TYPE(pim.pim_typever) != 7) {
|
|
/* not graft-ack */
|
|
s = dissect_pim_addr(&pd[offset], ep, pimv2_unicast, &advance);
|
|
if (s == NULL)
|
|
break;
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset, advance,
|
|
"Upstream-neighbor: %s", s);
|
|
offset += advance;
|
|
}
|
|
|
|
if (&pd[offset + 2] > ep)
|
|
break;
|
|
ngroup = pd[offset + 1];
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset + 1, 1,
|
|
"Groups: %u", pd[offset + 1]);
|
|
offset += 2;
|
|
|
|
if (&pd[offset + 2] > ep)
|
|
break;
|
|
if (PIM_TYPE(pim.pim_typever) != 7) {
|
|
/* not graft-ack */
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset, 2,
|
|
"Holdtime: %u%s", pntohs(&pd[offset]),
|
|
pntohs(&pd[offset]) == 0xffff ? " (infty)" : "");
|
|
}
|
|
offset += 2;
|
|
|
|
for (i = 0; i < ngroup; i++) {
|
|
if (&pd[offset] >= ep)
|
|
goto breakbreak3;
|
|
|
|
s = dissect_pim_addr(&pd[offset], ep, pimv2_group, &advance);
|
|
if (s == NULL)
|
|
goto breakbreak3;
|
|
tigroup = proto_tree_add_text(pimopt_tree, NullTVB, offset, advance,
|
|
"Group %d: %s", i, s);
|
|
grouptree = proto_item_add_subtree(tigroup, ett_pim);
|
|
offset += advance;
|
|
|
|
if (&pd[offset + 4] > ep)
|
|
goto breakbreak3;
|
|
njoin = pntohs(&pd[offset]);
|
|
nprune = pntohs(&pd[offset + 2]);
|
|
|
|
tisub = proto_tree_add_text(grouptree, NullTVB, offset, 2,
|
|
"Join: %d", njoin);
|
|
subtree = proto_item_add_subtree(tisub, ett_pim);
|
|
off = offset + 4;
|
|
for (j = 0; j < nprune; j++) {
|
|
s = dissect_pim_addr(&pd[off], ep, pimv2_source,
|
|
&advance);
|
|
if (s == NULL)
|
|
goto breakbreak3;
|
|
proto_tree_add_text(subtree, NullTVB, off, advance,
|
|
"IP address: %s", s);
|
|
off += advance;
|
|
}
|
|
|
|
tisub = proto_tree_add_text(grouptree, NullTVB, offset + 2, 2,
|
|
"Prune: %d", nprune);
|
|
subtree = proto_item_add_subtree(tisub, ett_pim);
|
|
for (j = 0; j < nprune; j++) {
|
|
s = dissect_pim_addr(&pd[off], ep, pimv2_source,
|
|
&advance);
|
|
if (s == NULL)
|
|
goto breakbreak3;
|
|
proto_tree_add_text(subtree, NullTVB, off, advance,
|
|
"IP address: %s", s);
|
|
off += advance;
|
|
}
|
|
}
|
|
breakbreak3:
|
|
break;
|
|
}
|
|
|
|
case 4: /* bootstrap */
|
|
{
|
|
const char *s;
|
|
int advance;
|
|
int i, j;
|
|
int frpcnt;
|
|
proto_tree *grouptree = NULL;
|
|
proto_item *tigroup;
|
|
|
|
offset += sizeof(struct pim);
|
|
|
|
if (END_OF_FRAME < 2)
|
|
break;
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset, 2,
|
|
"Fragment tag: 0x%04x", pntohs(&pd[offset]));
|
|
offset += 2;
|
|
|
|
if (END_OF_FRAME < 2)
|
|
break;
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset, 1,
|
|
"Hash mask len: %u", pd[offset]);
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset + 1, 1,
|
|
"BSR priority: %u", pd[offset + 1]);
|
|
offset += 2;
|
|
|
|
s = dissect_pim_addr(&pd[offset], &pd[offset + END_OF_FRAME],
|
|
pimv2_unicast, &advance);
|
|
if (s == NULL)
|
|
break;
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset, advance, "BSR: %s", s);
|
|
offset += advance;
|
|
|
|
for (i = 0; END_OF_FRAME > 0; i++) {
|
|
s = dissect_pim_addr(&pd[offset], &pd[offset + END_OF_FRAME],
|
|
pimv2_group, &advance);
|
|
if (s == NULL)
|
|
goto breakbreak4;
|
|
tigroup = proto_tree_add_text(pimopt_tree, NullTVB, offset, advance,
|
|
"Group %d: %s", i, s);
|
|
grouptree = proto_item_add_subtree(tigroup, ett_pim);
|
|
offset += advance;
|
|
|
|
if (END_OF_FRAME < 2)
|
|
goto breakbreak4;
|
|
proto_tree_add_text(grouptree, NullTVB, offset, 1,
|
|
"RP count: %u", pd[offset]);
|
|
proto_tree_add_text(grouptree, NullTVB, offset + 1, 1,
|
|
"FRP count: %u", pd[offset + 1]);
|
|
frpcnt = pd[offset + 1];
|
|
offset += 4;
|
|
|
|
for (j = 0; j < frpcnt && END_OF_FRAME > 0; j++) {
|
|
s = dissect_pim_addr(&pd[offset],
|
|
&pd[offset + END_OF_FRAME], pimv2_unicast, &advance);
|
|
if (s == NULL)
|
|
goto breakbreak4;
|
|
proto_tree_add_text(grouptree, NullTVB, offset, advance,
|
|
"RP %d: %s", j, s);
|
|
offset += advance;
|
|
|
|
if (END_OF_FRAME < 4)
|
|
goto breakbreak4;
|
|
proto_tree_add_text(grouptree, NullTVB, offset, 2,
|
|
"Holdtime: %u%s", pntohs(&pd[offset]),
|
|
pntohs(&pd[offset]) == 0xffff ? " (infty)" : "");
|
|
proto_tree_add_text(grouptree, NullTVB, offset + 3, 1,
|
|
"Priority: %u", pd[offset + 3]);
|
|
|
|
offset += 4;
|
|
}
|
|
}
|
|
|
|
breakbreak4:
|
|
break;
|
|
}
|
|
|
|
case 5: /* assert */
|
|
{
|
|
const char *s;
|
|
int advance;
|
|
|
|
offset += sizeof(struct pim);
|
|
|
|
s = dissect_pim_addr(&pd[offset], &pd[offset + END_OF_FRAME],
|
|
pimv2_group, &advance);
|
|
if (s == NULL)
|
|
break;
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset, advance, "Group: %s", s);
|
|
offset += advance;
|
|
|
|
s = dissect_pim_addr(&pd[offset], &pd[offset + END_OF_FRAME],
|
|
pimv2_unicast, &advance);
|
|
if (s == NULL)
|
|
break;
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset, advance, "Source: %s", s);
|
|
offset += advance;
|
|
|
|
if (END_OF_FRAME < 4)
|
|
break;
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset, 1, "%s",
|
|
decode_boolean_bitfield(pd[offset], 0x80, 8,
|
|
"RP Tree", "Not RP Tree"));
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset, 4, "Preference: %u",
|
|
pntohl(&pd[offset]) & 0x7fffffff);
|
|
offset += 4;
|
|
|
|
if (END_OF_FRAME < 4)
|
|
break;
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset, 4, "Metric: %u",
|
|
pntohl(&pd[offset]));
|
|
|
|
break;
|
|
}
|
|
|
|
case 8: /* Candidate-RP-Advertisement */
|
|
{
|
|
const char *s;
|
|
int advance;
|
|
int pfxcnt;
|
|
int i;
|
|
|
|
offset += sizeof(struct pim);
|
|
if (END_OF_FRAME < 4)
|
|
break;
|
|
pfxcnt = pd[offset];
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset, 1,
|
|
"Prefix-count: %u", pd[offset]);
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset + 1, 1,
|
|
"Priority: %u", pd[offset + 1]);
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset + 2, 2,
|
|
"Holdtime: %u%s", pntohs(&pd[offset + 2]),
|
|
pntohs(&pd[offset + 2]) == 0xffff ? " (infty)" : "");
|
|
offset += 4;
|
|
|
|
if (END_OF_FRAME <= 0)
|
|
break;
|
|
s = dissect_pim_addr(&pd[offset], &pd[offset + END_OF_FRAME],
|
|
pimv2_unicast, &advance);
|
|
if (s == NULL)
|
|
break;
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset, advance, "RP: %s", s);
|
|
offset += advance;
|
|
|
|
for (i = 0; i < pfxcnt && END_OF_FRAME > 0; i++) {
|
|
s = dissect_pim_addr(&pd[offset], &pd[offset + END_OF_FRAME],
|
|
pimv2_group, &advance);
|
|
if (s == NULL)
|
|
goto breakbreak8;
|
|
proto_tree_add_text(pimopt_tree, NullTVB, offset, advance,
|
|
"Group %d: %s", i, s);
|
|
offset += advance;
|
|
}
|
|
breakbreak8:
|
|
break;
|
|
}
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
done:;
|
|
}
|
|
|
|
void
|
|
proto_register_pim(void)
|
|
{
|
|
static hf_register_info hf[] = {
|
|
{ &hf_pim_version,
|
|
{ "Version", "pim.version",
|
|
FT_UINT8, BASE_DEC, NULL, 0x0, "" }},
|
|
{ &hf_pim_type,
|
|
{ "Type", "pim.type",
|
|
FT_UINT8, BASE_DEC, NULL, 0x0, "" }},
|
|
{ &hf_pim_cksum,
|
|
{ "Checksum", "pim.cksum",
|
|
FT_UINT16, BASE_HEX, NULL, 0x0, "" }},
|
|
};
|
|
static gint *ett[] = {
|
|
&ett_pim,
|
|
};
|
|
|
|
proto_pim = proto_register_protocol("Protocol Independent Multicast",
|
|
"pim");
|
|
proto_register_field_array(proto_pim, hf, array_length(hf));
|
|
proto_register_subtree_array(ett, array_length(ett));
|
|
}
|
|
|
|
void
|
|
proto_reg_handoff_pim(void)
|
|
{
|
|
old_dissector_add("ip.proto", IP_PROTO_PIM, dissect_pim);
|
|
}
|
|
|