/* packet-babel.c * Routines for Babel dissection (RFC 6126) * Copyright 2011 by Juliusz Chroboczek * * Wireshark - Network traffic analyzer * By Gerald Combs * Copyright 1998 Gerald Combs * * SPDX-License-Identifier: GPL-2.0-or-later */ #include "config.h" #include #include void proto_register_babel(void); void proto_reg_handoff_babel(void); static dissector_handle_t babel_handle; static int proto_babel; static int ett_babel; static int hf_babel_magic; static int hf_babel_version; static int hf_babel_bodylen; static int hf_babel_message; static int ett_message; static int hf_babel_message_type; static int hf_babel_message_length; static int hf_babel_message_nonce; static int hf_babel_message_interval; static int hf_babel_message_seqno; static int hf_babel_message_ae; static int hf_babel_message_prefix; static int hf_babel_message_rxcost; static int hf_babel_message_routerid; static int hf_babel_message_flags; static int hf_babel_message_plen; static int hf_babel_message_omitted; static int hf_babel_message_metric; static int hf_babel_message_hopcount; static int hf_babel_message_index; static int hf_babel_subtlv; static int hf_babel_subtlv_type; static int hf_babel_subtlv_len; static int hf_babel_subtlv_diversity; static int ett_subtree; static int ett_packet_trailer; static int ett_unicast; static int ett_subtlv; static int ett_timestamp; static int ett_mandatory; #define UDP_PORT_RANGE_BABEL "6696" #define MESSAGE_PAD1 0 #define MESSAGE_PADN 1 #define MESSAGE_ACK_REQ 2 #define MESSAGE_ACK 3 #define MESSAGE_HELLO 4 #define MESSAGE_IHU 5 #define MESSAGE_ROUTER_ID 6 #define MESSAGE_NH 7 #define MESSAGE_UPDATE 8 #define MESSAGE_REQUEST 9 #define MESSAGE_MH_REQUEST 10 #define MESSAGE_TS_PC 11 #define MESSAGE_HMAC_OLD 12 #define MESSAGE_SRC_UPDATE 13 #define MESSAGE_SRC_REQUEST 14 #define MESSAGE_SRC_SEQNO 15 #define MESSAGE_HMAC 16 #define MESSAGE_PC 17 #define MESSAGE_CHALLENGE_REQUEST 18 #define MESSAGE_CHALLENGE_REPLY 19 /** sub-TLVs */ #define MESSAGE_SUB_PAD1 0 #define MESSAGE_SUB_PADN 1 #define MESSAGE_SUB_DIVERSITY 2 #define MESSAGE_SUB_TIMESTAMP 3 /** mask for bits */ #define UNICAST_FLAG 0x80 #define MANDATORY_FLAG 128 /** message string values listed in rfc7557 */ static const value_string messages[] = { { MESSAGE_PAD1, "pad1"}, { MESSAGE_PADN, "padn"}, { MESSAGE_ACK_REQ, "ack-req"}, { MESSAGE_ACK, "ack"}, { MESSAGE_HELLO, "hello"}, { MESSAGE_IHU, "ihu"}, { MESSAGE_ROUTER_ID, "router-id"}, { MESSAGE_NH, "nh"}, { MESSAGE_UPDATE, "update"}, { MESSAGE_REQUEST, "request"}, { MESSAGE_MH_REQUEST, "mh-request"}, { MESSAGE_TS_PC, "ts/pc (obsolete)"}, { MESSAGE_HMAC_OLD, "hmac" }, { MESSAGE_SRC_UPDATE, "source-specific-update"}, { MESSAGE_SRC_REQUEST, "source-specific-req"}, { MESSAGE_SRC_SEQNO, "source-specific-seqno"}, { MESSAGE_HMAC, "hmac"}, { MESSAGE_PC, "pc"}, { MESSAGE_CHALLENGE_REQUEST, "challenge-request"}, { MESSAGE_CHALLENGE_REPLY, "challenge-reply"}, { 0, NULL} }; static const value_string subtlvs[] = { { MESSAGE_SUB_PAD1, "sub-pad1"}, { MESSAGE_SUB_PADN, "sub-padn"}, { MESSAGE_SUB_DIVERSITY, "diversity"}, { MESSAGE_SUB_TIMESTAMP, "timestamp"}, { 0, NULL} }; static const value_string aes[] = { { 0, "Wildcard" }, { 1, "IPv4" }, { 2, "IPv6" }, { 3, "Link-Local IPv6"}, { 0, NULL } }; /* The prefix for v6-mapped IPv4 addresses. Format_address below returns IPv4 addresses in that format. */ static const unsigned char v4prefix[16] = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xFF, 0xFF, 0, 0, 0, 0 }; /* The following two functions return ephemeral or constant strings, no need to call free. */ static const char * format_address(wmem_allocator_t *scope, const unsigned char *prefix) { address addr; if (prefix == NULL) return "corrupt"; else if (memcmp(prefix, v4prefix, 12) == 0) { addr.type = AT_IPv4; addr.len = 4; addr.data = prefix + 12; return address_to_str(scope, &addr); } else { addr.type = AT_IPv6; addr.len = 16; addr.data = prefix; return address_to_str(scope, &addr); } } static const char * format_prefix(wmem_allocator_t *scope, const unsigned char *prefix, unsigned char plen) { return wmem_strdup_printf(scope, "%s/%u", format_address(scope, prefix), plen); } static int network_prefix(int ae, int plen, unsigned int omitted, tvbuff_t *tvb, int offset, const unsigned char *dp, unsigned int len, unsigned char *p_r) { unsigned pb; unsigned char prefix[16]; int consumed = 0; if (plen >= 0) pb = (plen + 7) / 8; else if (ae == 1) pb = 4; else pb = 16; if (pb > 16) return -1; memset(prefix, 0, 16); switch(ae) { case 0: break; case 1: if (omitted > 4 || pb > 4 || (pb > omitted && len < pb - omitted)) return -1; memcpy(prefix, v4prefix, 12); if (omitted) { if (dp == NULL) return -1; memcpy(prefix, dp, 12 + omitted); } if (pb > omitted) { tvb_memcpy(tvb, prefix + 12 + omitted, offset, pb - omitted); consumed = pb - omitted; } break; case 2: if (omitted > 16 || (pb > omitted && len < pb - omitted)) return -1; if (omitted) { if (dp == NULL) return -1; memcpy(prefix, dp, omitted); } if (pb > omitted) { tvb_memcpy(tvb, prefix + omitted, offset, pb - omitted); consumed = pb - omitted; } break; case 3: if (pb > 8 && len < pb - 8) return -1; prefix[0] = 0xfe; prefix[1] = 0x80; if (pb > 8) { tvb_memcpy(tvb, prefix + 8, offset, pb - 8); consumed = pb - 8; } break; default: return -1; } memcpy(p_r, prefix, 16); return consumed; } static int network_address(int ae, tvbuff_t *tvb, int offset, unsigned int len, unsigned char *a_r) { return network_prefix(ae, -1, 0, tvb, offset, NULL, len, a_r); } static const char * format_timestamp(const uint32_t i) { static char buf[sizeof("0000.000000s")]; snprintf(buf, sizeof(buf), "%u.%06us", i / 1000000, i % 1000000); return buf; } static int dissect_babel_subtlvs(tvbuff_t * tvb, uint8_t type, uint16_t beg, uint16_t end, proto_tree *message_tree) { proto_tree *channel_tree = NULL; proto_item *sub_item; uint8_t subtype, sublen; int i = 0; while(beg < end) { proto_tree *subtlv_tree; subtype = tvb_get_uint8(tvb, beg); if (subtype != MESSAGE_SUB_PAD1) { sublen = tvb_get_uint8(tvb, beg+1); } else { sublen = 0; } sub_item = proto_tree_add_uint_format(message_tree, hf_babel_subtlv, tvb, beg, sublen + ((subtype == MESSAGE_SUB_PAD1) ? 1 : 2), subtype, "Sub TLV %s (%u)", val_to_str_const(subtype, subtlvs, "unknown"), subtype); subtlv_tree = proto_item_add_subtree(sub_item, ett_subtlv); proto_tree_add_item(subtlv_tree, hf_babel_subtlv_type, tvb, beg, 1, ENC_BIG_ENDIAN); if(subtype == MESSAGE_SUB_PAD1){ beg += 1; continue; } proto_tree_add_item(subtlv_tree, hf_babel_subtlv_len, tvb, beg+1, 1, ENC_BIG_ENDIAN); if ((MANDATORY_FLAG & subtype) != 0) { proto_tree_add_subtree_format(subtlv_tree, tvb, beg+2, sublen, ett_mandatory, NULL, "Mandatory"); } switch(subtype) { case MESSAGE_SUB_PADN: break; case MESSAGE_SUB_DIVERSITY: { i = 0; channel_tree = proto_tree_add_subtree_format(subtlv_tree, tvb, beg+2, 0, ett_subtlv, NULL, "Channel"); while(i < sublen) { proto_tree_add_item(channel_tree, hf_babel_subtlv_diversity, tvb, beg+2+i, 1, ENC_BIG_ENDIAN); i++; } } break; case MESSAGE_SUB_TIMESTAMP: { if (type == MESSAGE_HELLO) { uint32_t t1 = tvb_get_uint32(tvb, beg+2, ENC_BIG_ENDIAN); proto_tree_add_subtree_format(subtlv_tree, tvb, beg+2, sublen, ett_timestamp, NULL, "Timestamp : %s", format_timestamp(t1)); } else if (type == MESSAGE_IHU) { uint32_t t1 = tvb_get_uint32(tvb, beg+2, ENC_BIG_ENDIAN); uint32_t t2 = tvb_get_uint32(tvb, beg+6, ENC_BIG_ENDIAN); proto_tree_add_subtree_format(subtlv_tree, tvb, beg+2, sublen, ett_timestamp, NULL, "Timestamp origin : %s", format_timestamp(t1)); proto_tree_add_subtree_format(subtlv_tree, tvb, beg+6, sublen, ett_timestamp, NULL, "Timestamp receive: %s", format_timestamp(t2)); } else { proto_tree_add_subtree_format(subtlv_tree, tvb, beg+2, sublen, ett_timestamp, NULL, "Bogus"); } } break; } beg += (sublen+2); } return end-beg; } /* The following function is used to read the packet body and the packet trailer */ static int // NOLINTNEXTLINE(misc-no-recursion) dissect_babel_body(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, int offset, uint16_t bodylen) { proto_item *ti = NULL; unsigned char v4_prefix[16] = {0}, v6_prefix[16] = {0}; int i; i = offset; while (i-offset < bodylen) { uint8_t type, len = 0; uint16_t total_length; proto_tree *message_tree = NULL; int message = 4 + i; type = tvb_get_uint8(tvb, message); if (type == MESSAGE_PAD1) total_length = 1; else { len = tvb_get_uint8(tvb, message + 1); total_length = len + 2; } col_append_fstr(pinfo->cinfo, COL_INFO, " %s", val_to_str_const(type, messages, "unknown")); ti = proto_tree_add_uint_format(tree, hf_babel_message, tvb, message, total_length, type, "Message %s (%u)", val_to_str_const(type, messages, "unknown"), type); if (tree) { message_tree = proto_item_add_subtree(ti, ett_message); proto_tree_add_item(message_tree, hf_babel_message_type, tvb, message, 1, ENC_BIG_ENDIAN); } if (type == MESSAGE_PAD1) { i++; continue; } if (tree) { proto_tree_add_item(message_tree, hf_babel_message_length, tvb, message + 1, 1, ENC_BIG_ENDIAN); if (type == MESSAGE_PADN) { } else if (type == MESSAGE_ACK_REQ) { proto_tree_add_item(message_tree, hf_babel_message_nonce, tvb, message + 4, 2, ENC_BIG_ENDIAN); proto_tree_add_item(message_tree, hf_babel_message_interval, tvb, message + 6, 2, ENC_BIG_ENDIAN); } else if (type == MESSAGE_ACK) { proto_tree_add_item(message_tree, hf_babel_message_nonce, tvb, message + 2, 2, ENC_BIG_ENDIAN); } else if (type == MESSAGE_HELLO) { uint8_t unicast = tvb_get_uint8(tvb, 2); proto_tree_add_subtree_format(message_tree, tvb, message + 2, 2, ett_unicast, NULL, "Unicast : %u", unicast); proto_tree_add_item(message_tree, hf_babel_message_seqno, tvb, message + 4, 2, ENC_BIG_ENDIAN); proto_tree_add_item(message_tree, hf_babel_message_interval, tvb, message + 6, 2, ENC_BIG_ENDIAN); if(len > 6) dissect_babel_subtlvs(tvb, type, message + 8, message + 2 + len, message_tree); } else if (type == MESSAGE_IHU) { proto_tree *subtree; unsigned char addr_str[16]; int rc = network_address(tvb_get_uint8(tvb, message + 2), tvb, message + 8, len - 6, addr_str); proto_tree_add_item(message_tree, hf_babel_message_rxcost, tvb, message + 4, 2, ENC_BIG_ENDIAN); proto_tree_add_item(message_tree, hf_babel_message_interval, tvb, message + 6, 2, ENC_BIG_ENDIAN); subtree = proto_tree_add_subtree_format(message_tree, tvb, message + 4, len - 2, ett_subtree, NULL, "Address: %s", format_address(pinfo->pool, rc < 0 ? NULL : addr_str)); proto_tree_add_item(subtree, hf_babel_message_ae, tvb, message + 2, 1, ENC_BIG_ENDIAN); proto_tree_add_item(subtree, hf_babel_message_prefix, tvb, message + 4, len - 2, ENC_NA); if (rc < len - 6) dissect_babel_subtlvs(tvb, type, message + 8 + rc, message + 2 + len, message_tree); } else if (type == MESSAGE_ROUTER_ID) { proto_tree_add_item(message_tree, hf_babel_message_routerid, tvb, message + 4, 8, ENC_NA); } else if (type == MESSAGE_NH) { proto_tree *subtree; unsigned char nh[16]; int rc = network_address(tvb_get_uint8(tvb, message + 2), tvb, message + 4, len - 2, nh); subtree = proto_tree_add_subtree_format(message_tree, tvb, message + 4, len - 2, ett_subtree, NULL, "NH: %s", format_address(pinfo->pool, rc < 0 ? NULL : nh)); proto_tree_add_item(subtree, hf_babel_message_ae, tvb, message + 2, 1, ENC_BIG_ENDIAN); proto_tree_add_item(subtree, hf_babel_message_prefix, tvb, message + 4, len - 2, ENC_NA); } else if (type == MESSAGE_UPDATE) { proto_tree *subtree; unsigned char p[16]; uint8_t ae = tvb_get_uint8(tvb, message + 2); uint8_t flags = tvb_get_uint8(tvb, message + 3); uint8_t plen = tvb_get_uint8(tvb, message + 4); int rc = network_prefix(ae, plen, tvb_get_uint8(tvb, message + 5), tvb, message + 12, ae == 1 ? v4_prefix : v6_prefix, len - 10, p); if (rc >= 0 && (flags & 0x80)) { if (ae == 1) memcpy(v4_prefix, p, 16); else memcpy(v6_prefix, p, 16); } proto_tree_add_item(message_tree, hf_babel_message_flags, tvb, message + 3, 1, ENC_BIG_ENDIAN); proto_tree_add_item(message_tree, hf_babel_message_interval, tvb, message + 6, 2, ENC_BIG_ENDIAN); proto_tree_add_item(message_tree, hf_babel_message_seqno, tvb, message + 8, 2, ENC_BIG_ENDIAN); proto_tree_add_item(message_tree, hf_babel_message_metric, tvb, message + 10, 2, ENC_BIG_ENDIAN); subtree = proto_tree_add_subtree_format(message_tree, tvb, message + 12, len - 10, ett_subtree, NULL, "Prefix: %s", format_prefix(pinfo->pool, rc < 0 ? NULL : p, plen)); proto_tree_add_item(subtree, hf_babel_message_ae, tvb, message + 2, 1, ENC_BIG_ENDIAN); proto_tree_add_item(subtree, hf_babel_message_plen, tvb, message + 4, 1, ENC_BIG_ENDIAN); proto_tree_add_item(subtree, hf_babel_message_omitted, tvb, message + 5, 1, ENC_BIG_ENDIAN); proto_tree_add_item(subtree, hf_babel_message_prefix, tvb, message + 12, len - 10, ENC_NA); if (((uint8_t)rc) < len - 10) dissect_babel_subtlvs(tvb, type, message + 12 + rc, message + 2 + len, message_tree); } else if (type == MESSAGE_REQUEST) { proto_tree *subtree; unsigned char p[16]; uint8_t plen = tvb_get_uint8(tvb, message + 3); int rc = network_prefix(tvb_get_uint8(tvb, message + 2), plen, 0, tvb, message + 4, NULL, len - 2, p); subtree = proto_tree_add_subtree_format(message_tree, tvb, message + 4, len - 2, ett_subtree, NULL, "Prefix: %s", format_prefix(pinfo->pool, rc < 0 ? NULL : p, plen)); proto_tree_add_item(subtree, hf_babel_message_ae, tvb, message + 2, 1, ENC_BIG_ENDIAN); proto_tree_add_item(subtree, hf_babel_message_plen, tvb, message + 3, 1, ENC_BIG_ENDIAN); proto_tree_add_item(subtree, hf_babel_message_prefix, tvb, message + 4, len - 2, ENC_NA); } else if (type == MESSAGE_MH_REQUEST) { proto_tree *subtree; unsigned char p[16]; uint8_t plen = tvb_get_uint8(tvb, message + 3); int rc = network_prefix(tvb_get_uint8(tvb, message + 2), plen, 0, tvb, message + 16, NULL, len - 14, p); proto_tree_add_item(message_tree, hf_babel_message_seqno, tvb, message + 4, 2, ENC_BIG_ENDIAN); proto_tree_add_item(message_tree, hf_babel_message_hopcount, tvb, message + 6, 1, ENC_BIG_ENDIAN); proto_tree_add_item(message_tree, hf_babel_message_routerid, tvb, message + 8, 8, ENC_NA); subtree = proto_tree_add_subtree_format(message_tree, tvb, message + 16, len - 14, ett_subtree, NULL, "Prefix: %s", format_prefix(pinfo->pool, rc < 0 ? NULL : p, plen)); proto_tree_add_item(subtree, hf_babel_message_ae, tvb, message + 2, 1, ENC_BIG_ENDIAN); proto_tree_add_item(subtree, hf_babel_message_plen, tvb, message + 3, 1, ENC_BIG_ENDIAN); proto_tree_add_item(subtree, hf_babel_message_prefix, tvb, message + 16, len - 14, ENC_NA); } else if (type == MESSAGE_PC){ proto_tree_add_item(message_tree, hf_babel_message_index, tvb, message + 2, 4, ENC_NA); } } i += len + 2; } uint8_t packet_len = tvb_reported_length(tvb) - bodylen - 4; if ((offset == 0) && (packet_len != 0)) { proto_tree * subtree; subtree = proto_tree_add_subtree_format(tree, tvb, 4+bodylen, packet_len, ett_packet_trailer, NULL, "Packet Trailer (%u)", packet_len); increment_dissection_depth(pinfo); dissect_babel_body(tvb, pinfo, subtree, bodylen, packet_len); decrement_dissection_depth(pinfo); } return i; } static int dissect_babel(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void *data _U_) { proto_item *ti; proto_tree *babel_tree = NULL; uint8_t version; uint16_t bodylen; if (tvb_captured_length(tvb) < 4) return 0; if (tvb_get_uint8(tvb, 0) != 42) return 0; version = tvb_get_uint8(tvb, 1); col_set_str(pinfo->cinfo, COL_PROTOCOL, "Babel"); col_set_str(pinfo->cinfo, COL_INFO, "Babel"); if (version != 2) { col_add_fstr(pinfo->cinfo, COL_INFO, "Version %u", version); return 2; } if (tree) { ti = proto_tree_add_item(tree, proto_babel, tvb, 0, -1, ENC_NA); babel_tree = proto_item_add_subtree(ti, ett_babel); proto_tree_add_item(babel_tree, hf_babel_magic, tvb, 0, 1, ENC_BIG_ENDIAN); proto_tree_add_item(babel_tree, hf_babel_version, tvb, 1, 1, ENC_BIG_ENDIAN); proto_tree_add_item(babel_tree, hf_babel_bodylen, tvb, 2, 2, ENC_BIG_ENDIAN); } bodylen = tvb_get_ntohs(tvb, 2); return dissect_babel_body(tvb, pinfo, babel_tree, 0, bodylen); } void proto_register_babel(void) { static hf_register_info hf[] = { { &hf_babel_magic, { "Magic", "babel.magic", FT_UINT8, BASE_DEC, NULL, 0, "Magic value 42", HFILL } }, { &hf_babel_version, { "Version", "babel.version", FT_UINT8, BASE_DEC, NULL, 0, "Version of the Babel protocol", HFILL } }, { &hf_babel_bodylen, { "Body Length", "babel.bodylen", FT_UINT16, BASE_DEC, NULL, 0, NULL, HFILL } }, { &hf_babel_message, { "Message", "babel.message", FT_UINT8, BASE_DEC, NULL, 0, "Babel Message", HFILL } }, { &hf_babel_message_type, { "Message Type", "babel.message.type", FT_UINT8, BASE_DEC, VALS(messages), 0, NULL, HFILL } }, { &hf_babel_message_length, { "Message Length", "babel.message.length", FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL } }, { &hf_babel_message_nonce, { "Nonce", "babel.message.nonce", FT_UINT16, BASE_HEX, NULL, 0, NULL, HFILL } }, { &hf_babel_message_interval, { "Interval", "babel.message.interval", FT_UINT16, BASE_DEC, NULL, 0, "Interval (in centiseconds)", HFILL } }, { &hf_babel_message_seqno, { "Seqno", "babel.message.seqno", FT_UINT16, BASE_HEX, NULL, 0, NULL, HFILL } }, { &hf_babel_message_ae, { "Address Encoding", "babel.message.ae", FT_UINT8, BASE_DEC, VALS(aes), 0, NULL, HFILL } }, { &hf_babel_message_prefix, { "Raw Prefix", "babel.message.prefix", FT_BYTES, BASE_NONE, NULL, 0, NULL, HFILL } }, { &hf_babel_message_rxcost, { "Rxcost", "babel.message.rxcost", FT_UINT16, BASE_HEX, NULL, 0, "Rxcost (from the point of vue of the sender)", HFILL } }, { &hf_babel_message_routerid, { "Router ID", "babel.message.routerid", FT_BYTES, BASE_NONE, NULL, 0, NULL, HFILL } }, { &hf_babel_message_flags, { "Flags", "babel.message.flags", FT_UINT8, BASE_HEX, NULL, 0, NULL, HFILL } }, { &hf_babel_message_plen, { "Prefix Length", "babel.message.plen", FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL } }, { &hf_babel_message_omitted, { "Omitted Bytes", "babel.message.omitted", FT_UINT8, BASE_DEC, NULL, 0, "Number of bytes omitted from the prefix", HFILL } }, { &hf_babel_message_metric, { "Metric", "babel.message.metric", FT_UINT16, BASE_DEC, NULL, 0, NULL, HFILL } }, { &hf_babel_message_hopcount, { "Hop Count", "babel.message.hopcount", FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL } }, { &hf_babel_message_index, { "Index", "babel.message.index", FT_UINT32, BASE_DEC, NULL, 0, NULL, HFILL } }, { &hf_babel_subtlv, { "Sub-TLV", "babel.subtlv", FT_UINT8, BASE_DEC, NULL, 0, "Babel Sub-TLV", HFILL } }, { &hf_babel_subtlv_type, { "Sub-TLV Type", "babel.subtlv.type", FT_UINT8, BASE_DEC, VALS(subtlvs), 0, NULL, HFILL } }, { &hf_babel_subtlv_len, { "Sub-TLV Length", "babel.subtlv.length", FT_UINT8, BASE_DEC, VALS(subtlvs), 0, NULL, HFILL } }, { &hf_babel_subtlv_diversity, { "Channel", "babel.subtlv.diversity.channel", FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL } } }; static int *ett[] = { &ett_babel, &ett_message, &ett_subtree, &ett_packet_trailer, &ett_unicast, &ett_subtlv, &ett_timestamp, &ett_mandatory }; proto_babel = proto_register_protocol("Babel Routing Protocol", "Babel", "babel"); proto_register_field_array(proto_babel, hf, array_length(hf)); proto_register_subtree_array(ett, array_length(ett)); babel_handle = register_dissector("babel", dissect_babel, proto_babel); } void proto_reg_handoff_babel(void) { dissector_add_uint_range_with_preference("udp.port", UDP_PORT_RANGE_BABEL, babel_handle); } /* * Editor modelines - https://www.wireshark.org/tools/modelines.html * * Local variables: * c-basic-offset: 4 * tab-width: 8 * indent-tabs-mode: nil * End: * * vi: set shiftwidth=4 tabstop=8 expandtab: * :indentSize=4:tabSize=8:noTabs=true: */