wireshark/epan/dissectors/packet-lltd.c

967 lines
50 KiB
C

/* packet-lltd.c
* Routines for LLTD dissection
* Copyright 2012, Michael Mann
*
* Wireshark - Network traffic analyzer
* By Gerald Combs <gerald@wireshark.org>
* Copyright 1998 Gerald Combs
*
* SPDX-License-Identifier: GPL-2.0-or-later
*/
#include "config.h"
#include <epan/packet.h>
#include <epan/etypes.h>
#include <epan/expert.h>
void proto_register_lltd(void);
void proto_reg_handoff_lltd(void);
static int proto_lltd = -1;
static int hf_lltd_version = -1;
static int hf_lltd_type_of_service = -1;
static int hf_lltd_reserved = -1;
static int hf_lltd_discovery_func = -1;
static int hf_lltd_discovery_real_dest_addr = -1;
static int hf_lltd_discovery_real_src_addr = -1;
static int hf_lltd_discovery_xid = -1;
static int hf_lltd_discovery_seq_num = -1;
static int hf_lltd_discover_gen_num = -1;
static int hf_lltd_discover_num_stations = -1;
static int hf_lltd_discover_station = -1;
static int hf_lltd_hello_gen_num = -1;
static int hf_lltd_hello_current_address = -1;
static int hf_lltd_hello_apparent_address = -1;
static int hf_lltd_tlv_type = -1;
static int hf_lltd_tlv_length = -1;
static int hf_lltd_host_id = -1;
static int hf_lltd_char_p = -1;
static int hf_lltd_char_x = -1;
static int hf_lltd_char_f = -1;
static int hf_lltd_char_m = -1;
static int hf_lltd_char_l = -1;
static int hf_lltd_char_reserved = -1;
static int hf_lltd_physical_medium = -1;
static int hf_lltd_wireless_mode = -1;
static int hf_lltd_bssid = -1;
static int hf_lltd_ssid = -1;
static int hf_lltd_ipv4_address = -1;
static int hf_lltd_ipv6_address = -1;
static int hf_lltd_max_operation_rate = -1;
static int hf_lltd_performance_count_freq = -1;
static int hf_lltd_link_speed = -1;
static int hf_lltd_rssi = -1;
static int hf_lltd_machine_name = -1;
static int hf_lltd_support_info = -1;
static int hf_lltd_device_uuid = -1;
static int hf_lltd_qos_char_e = -1;
static int hf_lltd_qos_char_q = -1;
static int hf_lltd_qos_char_p = -1;
static int hf_lltd_qos_char_reserved = -1;
static int hf_lltd_80211_physical_medium = -1;
static int hf_lltd_sees_list_working_set = -1;
static int hf_lltd_repeater_ap_lineage = -1;
static int hf_lltd_emit_num_descs = -1;
static int hf_lltd_emit_type = -1;
static int hf_lltd_emit_pause = -1;
static int hf_lltd_emit_src_addr = -1;
static int hf_lltd_emit_dest_addr = -1;
static int hf_lltd_queryresp_more_descs = -1;
static int hf_lltd_queryresp_memory_descs = -1;
static int hf_lltd_queryresp_num_descs = -1;
static int hf_lltd_queryresp_type = -1;
static int hf_lltd_queryresp_real_src_addr = -1;
static int hf_lltd_queryresp_ethernet_src_addr = -1;
static int hf_lltd_queryresp_ethernet_dest_addr = -1;
static int hf_lltd_flat_crc_bytes = -1;
static int hf_lltd_flat_crc_packets = -1;
static int hf_lltd_query_large_tlv_type = -1;
static int hf_lltd_query_large_tlv_offset = -1;
static int hf_lltd_querylargeresp_more_descs = -1;
static int hf_lltd_querylargeresp_memory_descs = -1;
static int hf_lltd_querylargeresp_num_descs = -1;
static int hf_lltd_querylargeresp_data = -1;
static int hf_lltd_qos_diag_func = -1;
static int hf_lltd_qos_real_dest_addr = -1;
static int hf_lltd_qos_real_src_addr = -1;
static int hf_lltd_qos_seq_num = -1;
static int hf_lltd_qos_initialize_interrupt_mod = -1;
static int hf_lltd_qos_ready_sink_link_speed = -1;
static int hf_lltd_qos_ready_perf_count_freq = -1;
static int hf_lltd_qos_probe_controller_transmit_timestamp = -1;
static int hf_lltd_qos_probe_sink_receive_timestamp = -1;
static int hf_lltd_qos_probe_sink_transmit_timestamp = -1;
static int hf_lltd_qos_probe_test_type = -1;
static int hf_lltd_qos_probe_packet_id = -1;
static int hf_lltd_qos_probe_t = -1;
static int hf_lltd_qos_probe_8021p_value = -1;
static int hf_lltd_qos_probe_payload = -1;
static int hf_lltd_qos_error_value = -1;
static int hf_lltd_qos_count_snapshot_history = -1;
static int hf_lltd_qos_query_resp_r = -1;
static int hf_lltd_qos_query_resp_e = -1;
static int hf_lltd_qos_query_resp_num_events = -1;
static int hf_lltd_qos_query_resp_controller_timestamp = -1;
static int hf_lltd_qos_query_resp_sink_timestamp = -1;
static int hf_lltd_qos_query_resp_packet_id = -1;
static int hf_lltd_qos_query_resp_reserved = -1;
static int hf_lltd_qos_counter_result_subsec_span = -1;
static int hf_lltd_qos_counter_result_byte_scale = -1;
static int hf_lltd_qos_counter_result_packet_scale = -1;
static int hf_lltd_qos_counter_result_history_size = -1;
static int hf_lltd_qos_snapshot_bytes_recv = -1;
static int hf_lltd_qos_snapshot_packets_recv= -1;
static int hf_lltd_qos_snapshot_bytes_sent = -1;
static int hf_lltd_qos_snapshot_packets_sent= -1;
static gint ett_lltd = -1;
static gint ett_base_header = -1;
static gint ett_discover_stations = -1;
static gint ett_tlv = -1;
static gint ett_tlv_item = -1;
static gint ett_characteristics = -1;
static gint ett_qos_characteristics = -1;
static gint ett_repeater_ap_lineage = -1;
static gint ett_emitee_descs = -1;
static gint ett_emitee_descs_item = -1;
static gint ett_recvee_descs = -1;
static gint ett_recvee_descs_item = -1;
static gint ett_qos_event_descs = -1;
static gint ett_qos_event_item = -1;
static gint ett_qos_snapshot_list = -1;
static gint ett_qos_snapshot_item = -1;
static expert_field ei_lltd_tlv_length_invalid = EI_INIT;
static expert_field ei_lltd_too_many_paths = EI_INIT;
static expert_field ei_lltd_type_of_service = EI_INIT;
static expert_field ei_lltd_char_reserved = EI_INIT;
static expert_field ei_lltd_qos_seq_num = EI_INIT;
static expert_field ei_lltd_discovery_func = EI_INIT;
static expert_field ei_lltd_tlv_type = EI_INIT;
static expert_field ei_lltd_qos_diag_func = EI_INIT;
#define LLTD_CHARACTERISTIC_P_MASK 0x80000000
#define LLTD_CHARACTERISTIC_X_MASK 0x40000000
#define LLTD_CHARACTERISTIC_F_MASK 0x20000000
#define LLTD_CHARACTERISTIC_M_MASK 0x10000000
#define LLTD_CHARACTERISTIC_L_MASK 0x08000000
#define LLTD_CHARACTERISTIC_RESERVE_MASK 0x07FFFFFF
#define LLTD_QOS_CHARACTERISTIC_E_MASK 0x80000000
#define LLTD_QOS_CHARACTERISTIC_Q_MASK 0x40000000
#define LLTD_QOS_CHARACTERISTIC_P_MASK 0x20000000
#define LLTD_QOS_CHARACTERISTIC_RESERVE_MASK 0x1FFFFFFF
#define LLTD_QUERY_RESP_M_MASK 0x8000
#define LLTD_QUERY_RESP_E_MASK 0x4000
#define LLTD_QUERY_RESP_NUM_DESCS_MASK 0x3FFF
static const value_string lltd_tos_vals[] = {
{ 0, "Topology discovery" },
{ 1, "Quick discovery" },
{ 2, "QoS Diagnostics" },
{ 0, NULL }
};
static const value_string lltd_discovery_vals[] = {
{ 0x00, "Discover" },
{ 0x01, "Hello" },
{ 0x02, "Emit" },
{ 0x03, "Train" },
{ 0x04, "Probe" },
{ 0x05, "Ack" },
{ 0x06, "Query" },
{ 0x07, "QueryResp" },
{ 0x08, "Reset" },
{ 0x09, "Charge" },
{ 0x0A, "Flat" },
{ 0x0B, "QueryLargeTlv" },
{ 0x0C, "QueryLargeTlvResp" },
{ 0, NULL }
};
static const value_string lltd_qos_diag_vals[] = {
{ 0x00, "QosInitializeSink" },
{ 0x01, "QosReady" },
{ 0x02, "QosProbe" },
{ 0x03, "QosQuery" },
{ 0x04, "QosQueryResp" },
{ 0x05, "QosReset" },
{ 0x06, "QosError" },
{ 0x07, "QosAck" },
{ 0x08, "QosCounterSnapshot" },
{ 0x09, "QosCounterResult" },
{ 0x0A, "QosCounterLease" },
{ 0, NULL }
};
static const value_string lltd_tlv_type_vals[] = {
{ 0x00, "End of Property List" },
{ 0x01, "Host ID" },
{ 0x02, "Characteristics" },
{ 0x03, "Physical Medium" },
{ 0x04, "Wireless Mode" },
{ 0x05, "802.11 BSSID" },
{ 0x06, "802.11 SSID" },
{ 0x07, "IPv4 Address" },
{ 0x08, "IPv6 Address" },
{ 0x09, "802.11 Maximum Operation Rate" },
{ 0x0A, "Performance Counter Frequency" },
{ 0x0C, "Link Speed" },
{ 0x0D, "802.11 RSSI" },
{ 0x0E, "Icon Image" },
{ 0x0F, "Machine Name" },
{ 0x10, "Support Information" },
{ 0x11, "Friendly Name" },
{ 0x12, "Device UUID" },
{ 0x13, "Hardware ID" },
{ 0x14, "QoS Characteristics" },
{ 0x15, "802.11 Physical Medium" },
{ 0x16, "AP Association Table" },
{ 0x18, "Detailed Icon Image" },
{ 0x19, "Sees-List Working Set" },
{ 0x1A, "Component Table" },
{ 0x1B, "Repeater AP Lineage" },
{ 0x1C, "Repeater AP Table" },
{ 0, NULL }
};
static const value_string lltd_wireless_mode_vals[] = {
{ 0x00, "802.11 IBSS or ad-hoc mode" },
{ 0x01, "802.11 infrastructure mode" },
{ 0, NULL }
};
static const value_string lltd_80211_physical_medium_vals[] = {
{ 0x00, "Unknown" },
{ 0x01, "FHSS 2.4 GHz" },
{ 0x02, "DSSS 2.4 GHz" },
{ 0x03, "IR Baseband" },
{ 0x04, "OFDM 5 GHz" },
{ 0x05, "HRDSSS" },
{ 0x06, "ERP" },
{ 0, NULL }
};
static const value_string lltd_emit_type_vals[] = {
{ 0x00, "Train" },
{ 0x01, "Probe" },
{ 0, NULL }
};
static const value_string lltd_queryresp_type_vals[] = {
{ 0x00, "Probe" },
{ 0x01, "ARP" },
{ 0, NULL }
};
static const value_string lltd_query_large_tlv_type_vals[] = {
{ 0x0E, "Icon Image" },
{ 0x11, "Friendly Name" },
{ 0x13, "Hardware ID" },
{ 0x16, "AP Association Table" },
{ 0x18, "Detailed Icon Image" },
{ 0x1A, "Component Table" },
{ 0x1C, "Repeater AP Table" },
{ 0, NULL }
};
static const value_string lltd_interrupt_mod_vals[] = {
{ 0x00, "Disable interrupt moderation" },
{ 0x01, "Enable interrupt moderation" },
{ 0xFF, "Use existing interrupt moderation setting" },
{ 0, NULL }
};
static const value_string lltd_qos_probe_test_type_vals[] = {
{ 0x00, "Timed Probe" },
{ 0x01, "Probegap originating from the controller" },
{ 0x02, "Probegap originating from the sink" },
{ 0, NULL }
};
static const value_string lltd_qos_error_vals[] = {
{ 0x00, "Insufficient Resources" },
{ 0x01, "Busy. Try again later" },
{ 0x02, "Interrupt moderation not available" },
{ 0, NULL }
};
static const true_false_string tfs_full_half_duplex = { "Full Duplex", "Half Duplex" };
static int
dissect_lltd_tlv(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, guint32 offset, gboolean* end)
{
guint8 type, length = 0;
proto_item *tlv_item, *type_item;
proto_tree *tlv_tree, *type_tree;
guint32 i, temp32;
type = tvb_get_guint8(tvb, offset);
if (type == 0)
{
/* End of Property type doesn't have length */
tlv_tree = proto_tree_add_subtree(tree, tvb, offset, 1, ett_tlv_item, &tlv_item, "TLV Item (End of Property List)");
*end = TRUE;
}
else
{
length = tvb_get_guint8(tvb, offset+1);
tlv_tree = proto_tree_add_subtree_format(tree, tvb, offset, length+2, ett_tlv_item, &tlv_item,
"TLV Item (%s)", val_to_str(type, lltd_tlv_type_vals, "Unknown (0x%02x)"));
*end = FALSE;
}
proto_tree_add_item(tlv_tree, hf_lltd_tlv_type, tvb, offset, 1, ENC_NA);
if (type != 0)
proto_tree_add_item(tlv_tree, hf_lltd_tlv_length, tvb, offset+1, 1, ENC_NA);
if ((type != 0) && (length > tvb_reported_length_remaining(tvb, offset+2)))
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "TLV Length field too big");
*end = TRUE;
return 2;
}
switch(type)
{
case 0x00: /* End of Property List */
/* No data, no length field */
return 1;
case 0x01: /* Host ID */
if (length != 6)
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Invalid Host ID length");
}
proto_tree_add_item(tlv_tree, hf_lltd_host_id, tvb, offset+2, 6, ENC_NA);
break;
case 0x02: /* Characteristics */
if (length != 4)
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Characteristics length");
}
else
{
type_tree = proto_tree_add_subtree(tree, tvb, offset+2, 4, ett_characteristics, &type_item, "Characteristics");
proto_tree_add_item(type_tree, hf_lltd_char_p, tvb, offset+2, 4, ENC_BIG_ENDIAN);
proto_tree_add_item(type_tree, hf_lltd_char_x, tvb, offset+2, 4, ENC_BIG_ENDIAN);
proto_tree_add_item(type_tree, hf_lltd_char_f, tvb, offset+2, 4, ENC_BIG_ENDIAN);
proto_tree_add_item(type_tree, hf_lltd_char_m, tvb, offset+2, 4, ENC_BIG_ENDIAN);
proto_tree_add_item(type_tree, hf_lltd_char_l, tvb, offset+2, 4, ENC_BIG_ENDIAN);
proto_tree_add_item(type_tree, hf_lltd_char_reserved, tvb, offset+2, 4, ENC_BIG_ENDIAN);
if (tvb_get_ntohl(tvb, offset+2) & LLTD_CHARACTERISTIC_RESERVE_MASK)
expert_add_info(pinfo, type_item, &ei_lltd_char_reserved);
}
break;
case 0x03: /* Physical Medium */
if (length != 4)
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Invalid Physical Medium length");
}
proto_tree_add_item(tlv_tree, hf_lltd_physical_medium, tvb, offset+2, 4, ENC_BIG_ENDIAN);
break;
case 0x04: /* Wireless Mode */
if (length != 1)
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Invalid Wireless Mode length");
}
proto_tree_add_item(tlv_tree, hf_lltd_wireless_mode, tvb, offset+2, 1, ENC_BIG_ENDIAN);
break;
case 0x05: /* 802.11 BSSID */
if (length != 6)
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Invalid BSSID length");
}
proto_tree_add_item(tlv_tree, hf_lltd_bssid, tvb, offset+2, 6, ENC_NA);
break;
case 0x06: /* 802.11 SSID */
if (length > 32)
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "SSID length too large");
}
proto_tree_add_item(tlv_tree, hf_lltd_ssid, tvb, offset+2, length, ENC_NA|ENC_ASCII);
break;
case 0x07: /* IPv4 Address */
if (length != 4)
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Invalid IPv4 Address length");
}
proto_tree_add_item(tlv_tree, hf_lltd_ipv4_address, tvb, offset+2, 4, ENC_BIG_ENDIAN);
break;
case 0x08: /* IPv6 Address */
if (length != 16)
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Invalid IPv6 Address length");
}
proto_tree_add_item(tlv_tree, hf_lltd_ipv6_address, tvb, offset+2, 16, ENC_NA);
break;
case 0x09: /* 802.11 Maximum Operation Rate */
if (length != 2)
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Invalid Maximum Operation Rate length");
}
proto_tree_add_item(tlv_tree, hf_lltd_max_operation_rate, tvb, offset+2, 2, ENC_BIG_ENDIAN);
break;
case 0x0A: /* Performance Counter Frequency */
if (length != 8)
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Invalid Performance Counter Frequency length");
}
proto_tree_add_item(tlv_tree, hf_lltd_performance_count_freq, tvb, offset+2, 8, ENC_BIG_ENDIAN);
break;
case 0x0C: /* Link Speed */
if (length != 4)
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Invalid Link Speed length");
}
proto_tree_add_item(tlv_tree, hf_lltd_link_speed, tvb, offset+2, 4, ENC_BIG_ENDIAN);
break;
case 0x0D: /* 802.11 RSSI */
if (length != 4)
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Invalid RSSI length");
}
proto_tree_add_item(tlv_tree, hf_lltd_rssi, tvb, offset+2, 4, ENC_BIG_ENDIAN);
break;
case 0x0F: /* Machine Name */
if (length > 32)
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Machine Name length too large");
}
proto_tree_add_item(tlv_tree, hf_lltd_machine_name, tvb, offset+2, length, ENC_LITTLE_ENDIAN|ENC_UCS_2);
break;
case 0x10: /* Support Information */
if (length > 64)
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Support Information length too large");
}
proto_tree_add_item(tlv_tree, hf_lltd_support_info, tvb, offset+2, length, ENC_LITTLE_ENDIAN|ENC_UCS_2);
break;
case 0x11: /* Friendly Name */
if (length != 0)
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Invalid Friendly Name length");
break;
case 0x12: /* Device UUID */
if (length != 22)
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Invalid Device UUID length");
}
proto_tree_add_item(tlv_tree, hf_lltd_device_uuid, tvb, offset+2, 22, ENC_NA);
break;
case 0x13: /* Hardware ID */
if (length != 0)
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Invalid Hardware ID length");
break;
case 0x14: /* QoS Characteristics */
if (length != 4)
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "QoS Characteristics length");
}
else
{
type_tree = proto_tree_add_subtree(tlv_tree, tvb, offset+2, 4, ett_qos_characteristics, &type_item, "QoS Characteristics");
proto_tree_add_item(type_tree, hf_lltd_qos_char_e, tvb, offset+2, 4, ENC_BIG_ENDIAN);
proto_tree_add_item(type_tree, hf_lltd_qos_char_q, tvb, offset+2, 4, ENC_BIG_ENDIAN);
proto_tree_add_item(type_tree, hf_lltd_qos_char_p, tvb, offset+2, 4, ENC_BIG_ENDIAN);
proto_tree_add_item(type_tree, hf_lltd_qos_char_reserved, tvb, offset+2, 4, ENC_BIG_ENDIAN);
temp32 = tvb_get_ntohl(tvb, offset+2);
if (temp32 & LLTD_QOS_CHARACTERISTIC_RESERVE_MASK)
expert_add_info(pinfo, type_item, &ei_lltd_char_reserved);
}
break;
case 0x15: /* 802.11 Physical Medium */
if (length != 1)
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Invalid 802.11 Physical Medium length");
}
proto_tree_add_item(tlv_tree, hf_lltd_80211_physical_medium, tvb, offset+2, 1, ENC_BIG_ENDIAN);
break;
case 0x19: /* Sees-List Working Set */
if (length != 2)
{
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Invalid Sees-List Working Set length");
}
proto_tree_add_item(tlv_tree, hf_lltd_sees_list_working_set, tvb, offset+2, 2, ENC_BIG_ENDIAN);
break;
case 0x1B: /* Repeater AP Lineage */
type_tree = proto_tree_add_subtree(tree, tvb, offset+2, length, ett_repeater_ap_lineage, NULL, "Repeater AP Lineage");
for (i = 0; i < length; i += 6)
proto_tree_add_item(type_tree, hf_lltd_repeater_ap_lineage, tvb, offset+2+i, 6, ENC_NA);
if (length > 36)
expert_add_info(pinfo, tlv_item, &ei_lltd_too_many_paths);
break;
case 0x0E: /* Icon Image */
case 0x16: /* AP Association Table */
case 0x18: /* Detailed Icon Image */
case 0x1A: /* Component Table */
case 0x1C: /* Repeater AP Table */
if (length != 0)
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_length_invalid, "Invalid length");
break;
default:
expert_add_info_format(pinfo, tlv_item, &ei_lltd_tlv_type, "Invalid TLV Type 0x%02x", type);
break;
}
return length+2;
}
static void
dissect_lltd_discovery(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, int offset)
{
proto_item *header_item, *func_item;
proto_tree *header_tree, *func_tree, *func_subtree;
guint8 func;
guint16 temp16;
gboolean end_tlv = FALSE;
int loop_offset, start_offset;
func = tvb_get_guint8(tvb, offset);
proto_tree_add_item(tree, hf_lltd_discovery_func, tvb, offset, 1, ENC_NA);
col_add_fstr(pinfo->cinfo, COL_INFO, "%s", val_to_str(func, lltd_discovery_vals, "Unknown (0x%02x)"));
offset++;
/* Demultiplex header */
header_tree = proto_tree_add_subtree(tree, tvb, offset, 14, ett_base_header, &header_item, "Base header");
proto_tree_add_item(header_tree, hf_lltd_discovery_real_dest_addr, tvb, offset, 6, ENC_NA);
proto_tree_add_item(header_tree, hf_lltd_discovery_real_src_addr, tvb, offset+6, 6, ENC_NA);
if (func == 0)
proto_tree_add_item(header_tree, hf_lltd_discovery_xid, tvb, offset+12, 2, ENC_BIG_ENDIAN);
else
proto_tree_add_item(header_tree, hf_lltd_discovery_seq_num, tvb, offset+12, 2, ENC_BIG_ENDIAN);
switch(func)
{
case 0x00: /* Discover */
proto_tree_add_item(tree, hf_lltd_discover_gen_num, tvb, offset+14, 2, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_discover_num_stations, tvb, offset+16, 2, ENC_BIG_ENDIAN);
temp16 = tvb_get_ntohs(tvb, offset+16);
if (temp16 > 0)
{
func_tree = proto_tree_add_subtree(tree, tvb, offset+18, temp16*6, ett_discover_stations, NULL, "Stations");
for (loop_offset = 0; loop_offset < temp16*6; loop_offset += 6)
proto_tree_add_item(func_tree, hf_lltd_discover_station, tvb, offset+18+loop_offset, 6, ENC_NA);
}
break;
case 0x01: /* Hello */
proto_tree_add_item(tree, hf_lltd_hello_gen_num, tvb, offset+14, 2, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_hello_current_address, tvb, offset+16, 6, ENC_NA);
proto_tree_add_item(tree, hf_lltd_hello_apparent_address, tvb, offset+22, 6, ENC_NA);
func_tree = proto_tree_add_subtree(tree, tvb, offset+28, 0, ett_tlv, &func_item, "TLVs");
start_offset = loop_offset = offset+28;
while ((end_tlv == FALSE) && (tvb_reported_length_remaining(tvb, loop_offset) >= 1))
{
loop_offset += dissect_lltd_tlv(tvb, pinfo, func_tree, loop_offset, &end_tlv);
}
proto_item_set_len(func_item, loop_offset-start_offset);
break;
case 0x02: /* Emit */
proto_tree_add_item(tree, hf_lltd_emit_num_descs, tvb, offset+14, 2, ENC_BIG_ENDIAN);
temp16 = tvb_get_ntohs(tvb, offset+14);
if (temp16 > 0)
{
func_tree = proto_tree_add_subtree(tree, tvb, offset+16, temp16*14, ett_emitee_descs, NULL, "EmiteeDescs");
for (loop_offset = 0; loop_offset < temp16*14; loop_offset += 14)
{
func_subtree = proto_tree_add_subtree(func_tree, tvb, offset+16+loop_offset, 14, ett_emitee_descs_item, NULL, "EmiteeDescs Item");
proto_tree_add_item(func_subtree, hf_lltd_emit_type, tvb, offset+16+loop_offset, 1, ENC_NA);
proto_tree_add_item(func_subtree, hf_lltd_emit_pause, tvb, offset+16+loop_offset+1, 1, ENC_NA);
proto_tree_add_item(func_subtree, hf_lltd_emit_src_addr, tvb, offset+16+loop_offset+2, 6, ENC_NA);
proto_tree_add_item(func_subtree, hf_lltd_emit_dest_addr, tvb, offset+16+loop_offset+8, 6, ENC_NA);
}
}
break;
case 0x07: /* QueryResp */
proto_tree_add_item(tree, hf_lltd_queryresp_more_descs, tvb, offset+14, 2, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_queryresp_memory_descs, tvb, offset+14, 2, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_queryresp_num_descs, tvb, offset+14, 2, ENC_BIG_ENDIAN);
temp16 = tvb_get_ntohs(tvb, offset+14) & LLTD_QUERY_RESP_NUM_DESCS_MASK;
if (temp16 > 0)
{
func_tree = proto_tree_add_subtree(tree, tvb, offset+16, temp16*20, ett_recvee_descs, NULL, "RecveeDescs");
for (loop_offset = 0; loop_offset < temp16*14; loop_offset += 20)
{
func_subtree = proto_tree_add_subtree(func_tree, tvb, offset+16+loop_offset, 20,
ett_recvee_descs_item, NULL, "RecveeDescs Item");
proto_tree_add_item(func_subtree, hf_lltd_queryresp_type, tvb, offset+16+loop_offset, 2, ENC_BIG_ENDIAN);
proto_tree_add_item(func_subtree, hf_lltd_queryresp_real_src_addr, tvb, offset+16+loop_offset+2, 6, ENC_NA);
proto_tree_add_item(func_subtree, hf_lltd_queryresp_ethernet_src_addr, tvb, offset+16+loop_offset+8, 6, ENC_NA);
proto_tree_add_item(func_subtree, hf_lltd_queryresp_ethernet_dest_addr, tvb, offset+16+loop_offset+14, 6, ENC_NA);
}
}
break;
case 0x0A: /* Flat */
proto_tree_add_item(tree, hf_lltd_flat_crc_bytes, tvb, offset+14, 4, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_flat_crc_packets, tvb, offset+18, 1, ENC_BIG_ENDIAN);
break;
case 0x0B: /* QueryLargeTlv */
proto_tree_add_item(tree, hf_lltd_query_large_tlv_type, tvb, offset+14, 1, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_query_large_tlv_offset, tvb, offset+15, 3, ENC_BIG_ENDIAN);
break;
case 0x0C: /* QueryLargeTlvResp */
proto_tree_add_item(tree, hf_lltd_querylargeresp_more_descs, tvb, offset+14, 2, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_querylargeresp_memory_descs, tvb, offset+14, 2, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_querylargeresp_num_descs, tvb, offset+14, 2, ENC_BIG_ENDIAN);
temp16 = tvb_get_ntohs(tvb, offset+14) & LLTD_QUERY_RESP_NUM_DESCS_MASK;
if (temp16 > 0)
proto_tree_add_item(tree, hf_lltd_querylargeresp_data, tvb, offset+16, temp16, ENC_NA);
break;
case 0x03: /* Train */
case 0x04: /* Probe */
case 0x05: /* Ack */
case 0x06: /* Query */
case 0x08: /* Reset */
case 0x09: /* Charge */
/* No data */
break;
default:
expert_add_info_format(pinfo, header_item, &ei_lltd_discovery_func, "Invalid function 0x%02x", func);
break;
}
}
static void
dissect_lltd_qos(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, int offset)
{
proto_item *header_item;
proto_tree *header_tree, *func_tree, *func_subtree;
guint8 func;
guint16 seq_num, temp16;
int loop_offset;
func = tvb_get_guint8(tvb, offset);
proto_tree_add_item(tree, hf_lltd_qos_diag_func, tvb, offset, 1, ENC_NA);
col_add_fstr(pinfo->cinfo, COL_INFO, "%s", val_to_str(func, lltd_qos_diag_vals, "Unknown (0x%02x)"));
offset++;
header_tree = proto_tree_add_subtree(tree, tvb, offset, 14, ett_base_header, &header_item, "Base header");
proto_tree_add_item(header_tree, hf_lltd_qos_real_dest_addr, tvb, offset, 6, ENC_NA);
proto_tree_add_item(header_tree, hf_lltd_qos_real_src_addr, tvb, offset+6, 6, ENC_NA);
proto_tree_add_item(header_tree, hf_lltd_qos_seq_num, tvb, offset+12, 2, ENC_BIG_ENDIAN);
seq_num = tvb_get_ntohs(tvb, offset+12);
switch(func)
{
case 0x00: /* QosInitializeSink */
proto_tree_add_item(tree, hf_lltd_qos_initialize_interrupt_mod, tvb, offset+14, 1, ENC_BIG_ENDIAN);
break;
case 0x01: /* QosReady */
proto_tree_add_item(tree, hf_lltd_qos_ready_sink_link_speed, tvb, offset+14, 4, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_qos_ready_perf_count_freq, tvb, offset+18, 8, ENC_BIG_ENDIAN);
break;
case 0x02: /* QosProbe */
proto_tree_add_item(tree, hf_lltd_qos_probe_controller_transmit_timestamp, tvb, offset+14, 8, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_qos_probe_sink_receive_timestamp, tvb, offset+22, 8, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_qos_probe_sink_transmit_timestamp, tvb, offset+30, 8, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_qos_probe_test_type, tvb, offset+38, 1, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_qos_probe_packet_id, tvb, offset+39, 1, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_qos_probe_t, tvb, offset+40, 1, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_qos_probe_8021p_value, tvb, offset+40, 1, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_qos_probe_payload, tvb, offset+41, 5, ENC_NA);
break;
case 0x03: /* QosQuery */
case 0x07: /* QosAck */
if (seq_num == 0)
expert_add_info(pinfo, header_item, &ei_lltd_qos_seq_num);
/* No Data */
break;
case 0x04: /* QosQueryResp */
proto_tree_add_item(tree, hf_lltd_qos_query_resp_r, tvb, offset+14, 2, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_qos_query_resp_e, tvb, offset+14, 2, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_qos_query_resp_num_events, tvb, offset+14, 2, ENC_BIG_ENDIAN);
temp16 = tvb_get_ntohs(tvb, offset+14) & LLTD_QUERY_RESP_NUM_DESCS_MASK;
if (temp16 > 0)
{
func_tree = proto_tree_add_subtree(tree, tvb, offset+16, temp16*18, ett_qos_event_descs, NULL, "QosEventDesc");
for (loop_offset = 0; loop_offset < temp16*18; loop_offset += 18)
{
func_subtree = proto_tree_add_subtree(func_tree, tvb, offset+16+loop_offset, 18, ett_qos_event_item, NULL, "Qos Event");
proto_tree_add_item(func_subtree, hf_lltd_qos_query_resp_controller_timestamp, tvb, offset+16+loop_offset, 8, ENC_BIG_ENDIAN);
proto_tree_add_item(func_subtree, hf_lltd_qos_query_resp_sink_timestamp, tvb, offset+16+loop_offset+8, 8, ENC_BIG_ENDIAN);
proto_tree_add_item(func_subtree, hf_lltd_qos_query_resp_packet_id, tvb, offset+16+loop_offset+16, 1, ENC_BIG_ENDIAN);
proto_tree_add_item(func_subtree, hf_lltd_qos_query_resp_reserved, tvb, offset+16+loop_offset+17, 1, ENC_BIG_ENDIAN);
}
}
break;
case 0x05: /* QosReset */
/* No Data */
break;
case 0x06: /* QosError */
proto_tree_add_item(tree, hf_lltd_qos_error_value, tvb, offset+14, 2, ENC_BIG_ENDIAN);
break;
case 0x08: /* QosCounterSnapshot */
if (seq_num == 0)
expert_add_info(pinfo, header_item, &ei_lltd_qos_seq_num);
proto_tree_add_item(tree, hf_lltd_qos_count_snapshot_history, tvb, offset+14, 1, ENC_BIG_ENDIAN);
break;
case 0x09: /* QosCounterResult */
proto_tree_add_item(tree, hf_lltd_qos_counter_result_subsec_span, tvb, offset+14, 1, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_qos_counter_result_byte_scale, tvb, offset+15, 1, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_qos_counter_result_packet_scale, tvb, offset+16, 1, ENC_BIG_ENDIAN);
proto_tree_add_item(tree, hf_lltd_qos_counter_result_history_size, tvb, offset+17, 1, ENC_BIG_ENDIAN);
temp16 = tvb_get_guint8(tvb, offset+17);
if (temp16 > 0)
{
func_tree = proto_tree_add_subtree(tree, tvb, offset+18, temp16*4, ett_qos_snapshot_list, NULL, "Snapshot List");
for (loop_offset = 0; loop_offset < temp16*4; loop_offset += 4)
{
func_subtree = proto_tree_add_subtree(func_tree, tvb, offset+18+loop_offset, 4, ett_qos_snapshot_item, NULL, "Snapshot");
proto_tree_add_item(func_subtree, hf_lltd_qos_snapshot_bytes_recv, tvb, offset+16+loop_offset, 2, ENC_BIG_ENDIAN);
proto_tree_add_item(func_subtree, hf_lltd_qos_snapshot_packets_recv, tvb, offset+16+loop_offset+2, 2, ENC_BIG_ENDIAN);
proto_tree_add_item(func_subtree, hf_lltd_qos_snapshot_bytes_sent, tvb, offset+16+loop_offset+4, 2, ENC_BIG_ENDIAN);
proto_tree_add_item(func_subtree, hf_lltd_qos_snapshot_packets_sent, tvb, offset+16+loop_offset+6, 2, ENC_BIG_ENDIAN);
}
}
break;
case 0x0A: /* QosCounterLease */
/* No Data */
break;
default:
expert_add_info_format(pinfo, header_item, &ei_lltd_qos_diag_func, "Invalid function 0x%02x", func);
break;
}
}
static int
dissect_lltd(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void *data _U_)
{
proto_item *ti;
proto_tree *lltd_tree;
guint8 tos;
col_set_str(pinfo->cinfo, COL_PROTOCOL, "LLTD");
col_clear(pinfo->cinfo, COL_INFO);
ti = proto_tree_add_item(tree, proto_lltd, tvb, 0, -1, ENC_NA);
lltd_tree = proto_item_add_subtree(ti, ett_lltd);
proto_tree_add_item(lltd_tree, hf_lltd_version, tvb, 0, 1, ENC_NA);
proto_tree_add_item(lltd_tree, hf_lltd_type_of_service, tvb, 1, 1, ENC_NA);
tos = tvb_get_guint8(tvb, 1);
proto_tree_add_item(lltd_tree, hf_lltd_reserved, tvb, 2, 1, ENC_NA);
switch(tos)
{
case 0: /* Topology discovery */
case 1: /* Quick discovery */
dissect_lltd_discovery(tvb, pinfo, lltd_tree, 3);
break;
case 2: /* QoS Diagnostics */
dissect_lltd_qos(tvb, pinfo, lltd_tree, 3);
break;
default:
expert_add_info_format(pinfo, ti, &ei_lltd_type_of_service, "Invalid Type of Service value 0x%02x", tos);
break;
}
return tvb_captured_length(tvb);
}
void
proto_register_lltd(void)
{
static hf_register_info hf[] = {
{ &hf_lltd_version, {"Version", "lltd.version", FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_type_of_service, {"Type of Service", "lltd.tos", FT_UINT8, BASE_HEX, VALS(lltd_tos_vals), 0, NULL, HFILL }},
{ &hf_lltd_reserved, {"Reserved", "lltd.reserved", FT_UINT8, BASE_HEX, NULL, 0, NULL, HFILL }},
{ &hf_lltd_discovery_func, {"Discovery function", "lltd.discovery", FT_UINT8, BASE_HEX, VALS(lltd_discovery_vals), 0, NULL, HFILL }},
{ &hf_lltd_discovery_real_dest_addr, { "Real Destination Address", "lltd.discovery.real_dest_addr", FT_ETHER, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_discovery_real_src_addr, { "Real Source Address", "lltd.discovery.real_src_addr", FT_ETHER, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_discovery_xid, {"XID", "lltd.discovery.xid", FT_UINT16, BASE_HEX, NULL, 0, NULL, HFILL }},
{ &hf_lltd_discovery_seq_num, {"Sequence Number", "lltd.discovery.seq_num", FT_UINT16, BASE_HEX, NULL, 0, NULL, HFILL }},
{ &hf_lltd_discover_gen_num, {"Generation Number", "lltd.discover.gen_num", FT_UINT16, BASE_HEX, NULL, 0, NULL, HFILL }},
{ &hf_lltd_discover_num_stations, {"Number of Stations", "lltd.discover.num_stations", FT_UINT16, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_discover_station, { "Station", "lltd.discover.station", FT_ETHER, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_hello_gen_num, {"Generation Number", "lltd.hello.gen_num", FT_UINT16, BASE_HEX, NULL, 0, NULL, HFILL }},
{ &hf_lltd_hello_current_address, { "Current Mapper Address", "lltd.hello.current_address", FT_ETHER, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_hello_apparent_address, { "Current Apparent Address", "lltd.hello.apparent_address", FT_ETHER, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_tlv_type, {"Type", "lltd.tlv.type", FT_UINT8, BASE_HEX, VALS(lltd_tlv_type_vals), 0, NULL, HFILL }},
{ &hf_lltd_tlv_length, {"Length", "lltd.tlv.length", FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_host_id, { "Host ID", "lltd.host_id", FT_ETHER, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_char_p, {"Public NAT", "lltd.characteristic.public_nat", FT_BOOLEAN, 32, TFS(&tfs_true_false), LLTD_CHARACTERISTIC_P_MASK, NULL, HFILL }},
{ &hf_lltd_char_x, {"Private NAT", "lltd.characteristic.private_nat", FT_BOOLEAN, 32, TFS(&tfs_true_false), LLTD_CHARACTERISTIC_X_MASK, NULL, HFILL }},
{ &hf_lltd_char_f, {"Duplex", "lltd.characteristic.duplex", FT_BOOLEAN, 32, TFS(&tfs_full_half_duplex), LLTD_CHARACTERISTIC_F_MASK, NULL, HFILL }},
{ &hf_lltd_char_m, {"Management Web Page", "lltd.characteristic.web_page", FT_BOOLEAN, 32, TFS(&tfs_present_absent), LLTD_CHARACTERISTIC_M_MASK, NULL, HFILL }},
{ &hf_lltd_char_l, {"Looping Outbound Packets", "lltd.characteristic.loop", FT_BOOLEAN, 32, TFS(&tfs_true_false), LLTD_CHARACTERISTIC_L_MASK, NULL, HFILL }},
{ &hf_lltd_char_reserved, {"Reserved", "lltd.characteristic.reserved", FT_UINT32, BASE_HEX, NULL, LLTD_CHARACTERISTIC_RESERVE_MASK, NULL, HFILL }},
{ &hf_lltd_physical_medium, {"Physical Medium", "lltd.physical_medium", FT_UINT32, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_wireless_mode, {"Wireless Mode", "lltd.wireless_mode", FT_UINT8, BASE_HEX, VALS(lltd_wireless_mode_vals), 0, NULL, HFILL }},
{ &hf_lltd_bssid, { "BSSID", "lltd.bssid", FT_ETHER, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_ssid, { "SSID", "lltd.ssid", FT_STRING, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_ipv4_address, { "IPv4 Address", "lltd.ipv4_address", FT_IPv4, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_ipv6_address, { "IPv6 Address", "lltd.ipv6_address", FT_IPv6, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_max_operation_rate, {"Maximum Operational Rate (.5 Mbps)", "lltd.max_operation_rate", FT_UINT16, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_performance_count_freq, {"Performance Counter Frequency", "lltd.performance_count_freq", FT_UINT64, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_link_speed, {"Link Speed (100 bps)", "lltd.link_speed", FT_UINT32, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_rssi, {"RSSI", "lltd.rssi", FT_INT32, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_machine_name, { "Machine Name", "lltd.machine_name", FT_STRING /*FT_UCS2_LE */, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_support_info, { "Support Information", "lltd.support_info", FT_STRING /*FT_UCS2_LE */, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_device_uuid, { "Device UUID", "lltd.device_uuid", FT_GUID, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_char_e, {"Layer 2 Forwarding", "lltd.qos_characteristic.layer2_forwarding", FT_BOOLEAN, 32, TFS(&tfs_true_false), LLTD_QOS_CHARACTERISTIC_E_MASK, NULL, HFILL }},
{ &hf_lltd_qos_char_q, {"802.1q VLAN", "lltd.qos_characteristic.vlan", FT_BOOLEAN, 32, TFS(&tfs_supported_not_supported), LLTD_QOS_CHARACTERISTIC_Q_MASK, NULL, HFILL }},
{ &hf_lltd_qos_char_p, {"802.1q Priority Tagging", "lltd.qos_characteristic.tagging", FT_BOOLEAN, 32, TFS(&tfs_supported_not_supported), LLTD_QOS_CHARACTERISTIC_P_MASK, NULL, HFILL }},
{ &hf_lltd_qos_char_reserved, {"Reserved", "lltd.qos_characteristic.reserved", FT_UINT32, BASE_HEX, NULL, LLTD_QOS_CHARACTERISTIC_RESERVE_MASK, NULL, HFILL }},
{ &hf_lltd_80211_physical_medium, {"802.11 Physical Medium", "lltd.80211_physical_medium", FT_UINT8, BASE_HEX, VALS(lltd_80211_physical_medium_vals), 0, NULL, HFILL }},
{ &hf_lltd_sees_list_working_set, {"Sees-List Working Set", "lltd.sees_list_working_set", FT_UINT16, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_repeater_ap_lineage, { "Address Path to Root", "lltd.address_path_to_root", FT_ETHER, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_emit_num_descs, {"Number of EmiteeDescs", "lltd.emit.num_descs", FT_UINT16, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_emit_type, {"Type", "lltd.emit.type", FT_UINT8, BASE_HEX, VALS(lltd_emit_type_vals), 0, NULL, HFILL }},
{ &hf_lltd_emit_pause, {"Pause (ms)", "lltd.emit.pause", FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_emit_src_addr, { "Source Address", "lltd.emit.src_addr", FT_ETHER, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_emit_dest_addr, { "Destination Address", "lltd.emit.dest_addr", FT_ETHER, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_queryresp_more_descs, {"More RecveeDescs", "lltd.queryresp.more", FT_BOOLEAN, 16, TFS(&tfs_true_false), LLTD_QUERY_RESP_M_MASK, NULL, HFILL }},
{ &hf_lltd_queryresp_memory_descs, {"No memory left", "lltd.queryresp.memory", FT_BOOLEAN, 16, TFS(&tfs_true_false), LLTD_QUERY_RESP_E_MASK, NULL, HFILL }},
{ &hf_lltd_queryresp_num_descs, {"Number of RecveeDescs", "lltd.queryresp.num_descs", FT_UINT16, BASE_DEC, NULL, LLTD_QUERY_RESP_NUM_DESCS_MASK, NULL, HFILL }},
{ &hf_lltd_queryresp_type, {"Type", "lltd.queryresp.type", FT_UINT16, BASE_HEX, VALS(lltd_queryresp_type_vals), 0, NULL, HFILL }},
{ &hf_lltd_queryresp_real_src_addr, { "Real Source Address", "lltd.queryresp.real_src_addr", FT_ETHER, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_queryresp_ethernet_src_addr, { "Ethernet Source Address", "lltd.queryresp.ethernet_src_addr", FT_ETHER, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_queryresp_ethernet_dest_addr, { "Ethernet Destination Address", "lltd.queryresp.ethernet_dest_addr", FT_ETHER, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_flat_crc_bytes, {"Current Transmit Credit (bytes)", "lltd.flat.crc_bytes", FT_UINT32, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_flat_crc_packets, {"Current Transmit Credit (packets)", "lltd.flat.crc_packets", FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_query_large_tlv_type, {"Type", "lltd.query_large_tlv.type", FT_UINT8, BASE_HEX, VALS(lltd_query_large_tlv_type_vals), 0, NULL, HFILL }},
{ &hf_lltd_query_large_tlv_offset, {"Offset", "lltd.query_large_tlv.offset", FT_UINT24, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_querylargeresp_more_descs, {"More RecveeDescs", "lltd.querylargeresp.more", FT_BOOLEAN, 16, TFS(&tfs_true_false), LLTD_QUERY_RESP_M_MASK, NULL, HFILL }},
{ &hf_lltd_querylargeresp_memory_descs, {"No memory left", "lltd.querylargeresp.memory", FT_BOOLEAN, 16, TFS(&tfs_true_false), LLTD_QUERY_RESP_E_MASK, NULL, HFILL }},
{ &hf_lltd_querylargeresp_num_descs, {"Number of RecveeDescs", "lltd.querylargeresp.num_descs", FT_UINT16, BASE_DEC, NULL, LLTD_QUERY_RESP_NUM_DESCS_MASK, NULL, HFILL }},
{ &hf_lltd_querylargeresp_data, { "Data", "lltd.querylargeresp.data", FT_BYTES, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_diag_func, {"QoS Diagnostics function", "lltd.qos_diag", FT_UINT8, BASE_HEX, VALS(lltd_qos_diag_vals), 0, NULL, HFILL }},
{ &hf_lltd_qos_real_dest_addr, { "Real Destination Address", "lltd.qos.real_dest_addr", FT_ETHER, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_real_src_addr, { "Real Source Address", "lltd.qos.real_src_addr", FT_ETHER, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_seq_num, {"Sequence Number", "lltd.qos.seq_num", FT_UINT16, BASE_HEX, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_initialize_interrupt_mod, {"Interrupt Mod", "lltd.qos_initialize.interrupt_mod", FT_UINT8, BASE_HEX, VALS(lltd_interrupt_mod_vals), 0, NULL, HFILL }},
{ &hf_lltd_qos_ready_sink_link_speed, {"Sink Link Speed (100 bps)", "lltd.qos_ready.sink_link_speed", FT_UINT32, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_ready_perf_count_freq, {"Performance Counter Frequency", "lltd.qos_ready.performance_count_freq", FT_UINT64, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_probe_controller_transmit_timestamp, {"Controller Transmit Timestamp", "lltd.qos_probe.controller_transmit_timestamp", FT_UINT64, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_probe_sink_receive_timestamp, {"Sink Receive Timestamp", "lltd.qos_probe.sink_receive_timestamp", FT_UINT64, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_probe_sink_transmit_timestamp, {"Sink Transmit Timestamp", "lltd.qos_probe.sink_transmit_timestamp", FT_UINT64, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_probe_test_type, {"Test Type", "lltd.qos_probe.test_type", FT_UINT8, BASE_HEX, VALS(lltd_qos_probe_test_type_vals), 0, NULL, HFILL }},
{ &hf_lltd_qos_probe_packet_id, {"Packet ID", "lltd.qos_probe.packet_id", FT_UINT8, BASE_HEX, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_probe_t, {"802.1p Tag", "lltd.qos_probe.tag", FT_BOOLEAN, 8, TFS(&tfs_present_not_present), 0x80, NULL, HFILL }},
{ &hf_lltd_qos_probe_8021p_value, {"802.1p Value", "lltd.qos_probe.value", FT_UINT8, BASE_DEC, NULL, 0x7F, NULL, HFILL }},
{ &hf_lltd_qos_probe_payload, {"Payload", "lltd.qos_probe.payload", FT_BYTES, BASE_NONE, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_error_value, {"Error Code", "lltd.qos_error", FT_UINT16, BASE_DEC, VALS(lltd_qos_error_vals), 0, NULL, HFILL }},
{ &hf_lltd_qos_count_snapshot_history, {"History Size", "lltd.qos_count_snapshot.history", FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_query_resp_r, {"Receipt", "lltd.qos_query_resp.receipt", FT_BOOLEAN, 16, TFS(&tfs_true_false), LLTD_QUERY_RESP_M_MASK, NULL, HFILL }},
{ &hf_lltd_qos_query_resp_e, {"No memory left", "lltd.qos_query_resp.memory", FT_BOOLEAN, 16, TFS(&tfs_true_false), LLTD_QUERY_RESP_E_MASK, NULL, HFILL }},
{ &hf_lltd_qos_query_resp_num_events, {"Number of Events", "lltd.qos_query_resp.num_events", FT_UINT16, BASE_DEC, NULL, LLTD_QUERY_RESP_NUM_DESCS_MASK, NULL, HFILL }},
{ &hf_lltd_qos_query_resp_controller_timestamp, {"Controller Transmit Timestamp", "lltd.qos_query_resp.controller_timestamp", FT_UINT64, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_query_resp_sink_timestamp, {"Sink Receive Timestamp", "lltd.qos_query_resp.sink_timestamp", FT_UINT64, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_query_resp_packet_id, {"Packet ID", "lltd.qos_query_resp.packet_id", FT_UINT8, BASE_HEX, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_query_resp_reserved, {"Reserved", "lltd.qos_query_resp.reserved", FT_UINT8, BASE_HEX, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_counter_result_subsec_span, {"Subsecond Span (1/256th sec)", "lltd.qos_counter_result.subsec_span", FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_counter_result_byte_scale, {"Byte Scale (kb)", "lltd.qos_counter_result.byte_scale", FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_counter_result_packet_scale, {"Packet Scale", "lltd.qos_counter_result.packet_scale", FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_counter_result_history_size, {"History Size", "lltd.qos_counter_result.history_size", FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_snapshot_bytes_recv, {"Bytes Received", "lltd.qos_snapshot.bytes_recv", FT_UINT16, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_snapshot_packets_recv, {"Packets Received", "lltd.qos_snapshot.packets_recv", FT_UINT16, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_snapshot_bytes_sent, {"Bytes Sent", "lltd.qos_snapshot.bytes_sent", FT_UINT16, BASE_DEC, NULL, 0, NULL, HFILL }},
{ &hf_lltd_qos_snapshot_packets_sent, {"Packets Sent", "lltd.qos_snapshot.packets_sent", FT_UINT16, BASE_DEC, NULL, 0, NULL, HFILL }}
};
static gint *ett[] = {
&ett_lltd,
&ett_base_header,
&ett_discover_stations,
&ett_tlv,
&ett_tlv_item,
&ett_characteristics,
&ett_qos_characteristics,
&ett_repeater_ap_lineage,
&ett_emitee_descs,
&ett_emitee_descs_item,
&ett_recvee_descs,
&ett_recvee_descs_item,
&ett_qos_event_descs,
&ett_qos_event_item,
&ett_qos_snapshot_list,
&ett_qos_snapshot_item
};
static ei_register_info ei[] = {
{ &ei_lltd_tlv_length_invalid, { "lltd.tlv.length.invalid", PI_MALFORMED, PI_ERROR, "Invalid length", EXPFILL }},
{ &ei_lltd_char_reserved, { "lltd.characteristic.reserved.not_zero", PI_PROTOCOL, PI_WARN, "Non zero reserve bits", EXPFILL }},
{ &ei_lltd_too_many_paths, { "lltd.too_many_paths", PI_PROTOCOL, PI_WARN, "Too many paths to root", EXPFILL }},
{ &ei_lltd_tlv_type, { "lltd.tlv.type.invalid", PI_PROTOCOL, PI_WARN, "Invalid TLV Type 0x%02x", EXPFILL }},
{ &ei_lltd_discovery_func, { "lltd.discovery.invalid", PI_PROTOCOL, PI_WARN, "Invalid function 0x%02x", EXPFILL }},
{ &ei_lltd_qos_seq_num, { "lltd.qos.seq_num.cannot_be_zero", PI_PROTOCOL, PI_WARN, "Sequence number can not be 0", EXPFILL }},
{ &ei_lltd_qos_diag_func, { "lltd.qos_diag.invalid", PI_PROTOCOL, PI_WARN, "Invalid function 0x%02x", EXPFILL }},
{ &ei_lltd_type_of_service, { "lltd.tos.invalid", PI_PROTOCOL, PI_WARN, "Invalid Type of Service value 0x%02x", EXPFILL }},
};
expert_module_t* expert_lltd;
proto_lltd = proto_register_protocol("Link Layer Topology Discovery", "LLTD", "lltd");
proto_register_field_array(proto_lltd, hf, array_length(hf));
proto_register_subtree_array(ett, array_length(ett));
expert_lltd = expert_register_protocol(proto_lltd);
expert_register_field_array(expert_lltd, ei, array_length(ei));
}
void
proto_reg_handoff_lltd(void)
{
dissector_handle_t lltd_handle;
lltd_handle = create_dissector_handle(dissect_lltd, proto_lltd);
dissector_add_uint("ethertype", ETHERTYPE_LLTD, lltd_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:
*/