diff options
Diffstat (limited to 'epan/dissectors/packet-awdl.c')
-rw-r--r-- | epan/dissectors/packet-awdl.c | 2674 |
1 files changed, 2674 insertions, 0 deletions
diff --git a/epan/dissectors/packet-awdl.c b/epan/dissectors/packet-awdl.c new file mode 100644 index 00000000..fa90cb9f --- /dev/null +++ b/epan/dissectors/packet-awdl.c @@ -0,0 +1,2674 @@ +/* packet-awdl.c + * Routines for Apple Wireless Direct Link (AWDL) dissection + * + * Copyright 2017 David Kreitschmann <dkreitschmann@seemoo.tu-darmstadt.de> + * Copyright 2018 Milan Stute <mstute@seemoo.tu-darmstadt.de> + * + * Released as part of: + * Milan Stute, David Kreitschmann, and Matthias Hollick. "One Billion Apples' + * Secret Sauce: Recipe for the Apple Wireless Direct Link Ad hoc Protocol" + * in ACM MobiCom '18. https://doi.org/10.1145/3241539.3241566 + * + * SPDX-License-Identifier: GPL-2.0-or-later + */ + +#include <config.h> + +#include <math.h> + +#include <epan/packet.h> +#include <epan/expert.h> +#include "packet-llc.h" +#include "packet-ieee80211.h" +#include "packet-dns.h" +#include <epan/oui.h> + +void proto_register_awdl(void); +void proto_reg_handoff_awdl(void); + +static dissector_handle_t awdl_action_handle; +static dissector_handle_t awdl_data_handle; + +typedef struct awdl_tagged_field_data +{ + proto_item* item_tag; + proto_item* item_tag_length; +} awdl_tagged_field_data_t; + +static const unit_name_string units_ieee80211_tu = { " TU", NULL }; /* 1 TU = 1024 microseconds */ + +static int proto_awdl = -1; +static int proto_awdl_data = -1; + +static int hf_awdl_unknown = -1; + +static int hf_awdl_fixed_parameters = -1; +static int hf_awdl_tagged_parameters = -1; + +static int hf_awdl_data_seq = -1; +static int hf_awdl_data_header = -1; +static int hf_awdl_data_ethertype = -1; + +static int hf_awdl_type = -1; +static int hf_awdl_subtype = -1; +static int hf_awdl_rsvd = -1; +static int hf_awdl_phytime = -1; +static int hf_awdl_targettime = -1; +static int hf_awdl_txdelay = -1; + +static int hf_awdl_tag = -1; +static int hf_awdl_tag_number = -1; +static int hf_awdl_tag_length = -1; +static int hf_awdl_tag_data = -1; +static int hf_awdl_tag_padding = -1; + +static int hf_awdl_version = -1; +static int hf_awdl_version_minor = -1; +static int hf_awdl_version_major = -1; +static int hf_awdl_version_devclass = -1; + +static int hf_awdl_datastate_flags = -1; +static int hf_awdl_datastate_flags_0 = -1; +static int hf_awdl_datastate_flags_1 = -1; +static int hf_awdl_datastate_flags_2 = -1; +static int hf_awdl_datastate_flags_3 = -1; +static int hf_awdl_datastate_flags_4 = -1; +static int hf_awdl_datastate_flags_5 = -1; +static int hf_awdl_datastate_flags_6 = -1; +static int hf_awdl_datastate_flags_7 = -1; +static int hf_awdl_datastate_flags_8 = -1; +static int hf_awdl_datastate_flags_9 = -1; +static int hf_awdl_datastate_flags_10 = -1; +static int hf_awdl_datastate_flags_11 = -1; +static int hf_awdl_datastate_flags_12 = -1; +static int hf_awdl_datastate_flags_13 = -1; +static int hf_awdl_datastate_flags_14 = -1; +static int hf_awdl_datastate_flags_15 = -1; +static int hf_awdl_datastate_extflags = -1; +static int hf_awdl_datastate_extflags_0 = -1; +static int hf_awdl_datastate_extflags_1 = -1; +static int hf_awdl_datastate_extflags_2 = -1; +static int hf_awdl_datastate_extflags_3to15 = -1; +static int hf_awdl_datastate_infra_channel = -1; +static int hf_awdl_datastate_countrycode = -1; +static int hf_awdl_datastate_social_channel = -1; +static int hf_awdl_datastate_social_channel_map = -1; +static int hf_awdl_datastate_social_channel_map_6 = -1; +static int hf_awdl_datastate_social_channel_map_44 = -1; +static int hf_awdl_datastate_social_channel_map_149 = -1; +static int hf_awdl_datastate_social_channel_map_unused = -1; +static int hf_awdl_datastate_infra_bssid = -1; +static int hf_awdl_datastate_infra_address = -1; +static int hf_awdl_datastate_awdl_address = -1; +static int hf_awdl_datastate_umi = -1; +static int hf_awdl_datastate_umioptions = -1; +static int hf_awdl_datastate_umioptions_length = -1; +static int hf_awdl_datastate_logtrigger = -1; +static int hf_awdl_datastate_undecoded = -1; + +static int hf_awdl_synctree_addr = -1; + +static int hf_awdl_syncparams_master = -1; +static int hf_awdl_syncparams_awcounter = -1; +static int hf_awdl_syncparams_apbeaconalignment = -1; +static int hf_awdl_syncparams_tx_chan = -1; +static int hf_awdl_syncparams_tx_counter = -1; +static int hf_awdl_syncparams_master_chan = -1; +static int hf_awdl_syncparams_guard_time = -1; +static int hf_awdl_syncparams_aw_period = -1; +static int hf_awdl_syncparams_action_frame_period = -1; +static int hf_awdl_syncparams_awdl_flags = -1; +static int hf_awdl_syncparams_aw_ext_length = -1; +static int hf_awdl_syncparams_aw_cmn_length = -1; +static int hf_awdl_syncparams_aw_remaining = -1; +static int hf_awdl_syncparams_ext_min = -1; +static int hf_awdl_syncparams_ext_max_multi = -1; +static int hf_awdl_syncparams_ext_max_uni = -1; +static int hf_awdl_syncparams_ext_max_af = -1; +static int hf_awdl_syncparams_presence_mode = -1; + +static int hf_awdl_channelseq_enc = -1; +static int hf_awdl_channelseq_duplicate = -1; +static int hf_awdl_channelseq_step_count = -1; +static int hf_awdl_channelseq_fill_chan = -1; +static int hf_awdl_channelseq_channel_count = -1; +static int hf_awdl_channelseq_channel_list = -1; +static int hf_awdl_channelseq_channel = -1; +static int hf_awdl_channelseq_channel_number = -1; +static int hf_awdl_channelseq_channel_flags = -1; +static int hf_awdl_channelseq_channel_operating_class = -1; +/* legacy encoding flags */ +static int hf_awdl_channelseq_legacy_unused = -1; +static int hf_awdl_channelseq_legacy_band = -1; +static int hf_awdl_channelseq_legacy_bandwidth = -1; +static int hf_awdl_channelseq_legacy_control_channel = -1; + +static int hf_awdl_electionparams_master = -1; +static int hf_awdl_electionparams_flags = -1; +static int hf_awdl_electionparams_id = -1; +static int hf_awdl_electionparams_distance = -1; +static int hf_awdl_electionparams_mastermetric = -1; +static int hf_awdl_electionparams_selfmetric = -1; +static int hf_awdl_electionparams_unknown = -1; +static int hf_awdl_electionparams_private_master = -1; +static int hf_awdl_electionparams_private_mastermetric = -1; +static int hf_awdl_electionparams_private_id = -1; +static int hf_awdl_electionparams_private_phc = -1; + +static int hf_awdl_electionparams2_master = -1; +static int hf_awdl_electionparams2_other = -1; +static int hf_awdl_electionparams2_mastermetric = -1; +static int hf_awdl_electionparams2_selfmetric = -1; +static int hf_awdl_electionparams2_mastercounter = -1; +static int hf_awdl_electionparams2_selfcounter = -1; +static int hf_awdl_electionparams2_distance = -1; +static int hf_awdl_electionparams2_unknown = -1; +static int hf_awdl_electionparams2_reserved = -1; + +static int hf_awdl_dns_name_len = -1; +static int hf_awdl_dns_name = -1; +static int hf_awdl_dns_name_label = -1; +static int hf_awdl_dns_name_short = -1; +static int hf_awdl_dns_type = -1; +static int hf_awdl_dns_data_len = -1; +static int hf_awdl_dns_txt = -1; +static int hf_awdl_dns_ptr = -1; +static int hf_awdl_dns_ptr_label = -1; +static int hf_awdl_dns_ptr_short = -1; +static int hf_awdl_dns_target = -1; +static int hf_awdl_dns_target_label = -1; +static int hf_awdl_dns_target_short = -1; +static int hf_awdl_dns_unknown = -1; +static int hf_awdl_dns_priority = -1; +static int hf_awdl_dns_weight = -1; +static int hf_awdl_dns_port = -1; + +static int hf_awdl_serviceparams_sui = -1; +static int hf_awdl_serviceparams_enc_values = -1; +static int hf_awdl_serviceparams_bitmask = -1; +static int hf_awdl_serviceparams_bitmask_0 = -1; +static int hf_awdl_serviceparams_bitmask_1 = -1; +static int hf_awdl_serviceparams_bitmask_2 = -1; +static int hf_awdl_serviceparams_bitmask_3 = -1; +static int hf_awdl_serviceparams_bitmask_4 = -1; +static int hf_awdl_serviceparams_bitmask_5 = -1; +static int hf_awdl_serviceparams_bitmask_6 = -1; +static int hf_awdl_serviceparams_bitmask_7 = -1; +static int hf_awdl_serviceparams_bitmask_8 = -1; +static int hf_awdl_serviceparams_bitmask_9 = -1; +static int hf_awdl_serviceparams_bitmask_10 = -1; +static int hf_awdl_serviceparams_bitmask_11 = -1; +static int hf_awdl_serviceparams_bitmask_12= -1; +static int hf_awdl_serviceparams_bitmask_13 = -1; +static int hf_awdl_serviceparams_bitmask_14 = -1; +static int hf_awdl_serviceparams_bitmask_15 = -1; +static int hf_awdl_serviceparams_bitmask_16 = -1; +static int hf_awdl_serviceparams_bitmask_17 = -1; +static int hf_awdl_serviceparams_bitmask_18 = -1; +static int hf_awdl_serviceparams_bitmask_19 = -1; +static int hf_awdl_serviceparams_bitmask_20 = -1; +static int hf_awdl_serviceparams_bitmask_21 = -1; +static int hf_awdl_serviceparams_bitmask_22 = -1; +static int hf_awdl_serviceparams_bitmask_23 = -1; +static int hf_awdl_serviceparams_bitmask_24 = -1; +static int hf_awdl_serviceparams_bitmask_25 = -1; +static int hf_awdl_serviceparams_bitmask_26 = -1; +static int hf_awdl_serviceparams_bitmask_27 = -1; +static int hf_awdl_serviceparams_bitmask_28 = -1; +static int hf_awdl_serviceparams_bitmask_29 = -1; +static int hf_awdl_serviceparams_bitmask_30 = -1; +static int hf_awdl_serviceparams_bitmask_31 = -1; +static int hf_awdl_serviceparams_values = -1; +static int hf_awdl_serviceparams_values_0 = -1; +static int hf_awdl_serviceparams_values_1 = -1; +static int hf_awdl_serviceparams_values_2 = -1; +static int hf_awdl_serviceparams_values_3 = -1; +static int hf_awdl_serviceparams_values_4 = -1; +static int hf_awdl_serviceparams_values_5 = -1; +static int hf_awdl_serviceparams_values_6 = -1; +static int hf_awdl_serviceparams_values_7 = -1; + +static int hf_awdl_arpa = -1; +static int hf_awdl_arpa_flags = -1; +static int hf_awdl_arpa_name = -1; +static int hf_awdl_arpa_short = -1; + +static int hf_awdl_ht_unknown = -1; +/* from hf_ieee80211_* from packet-ieee80211.c */ +static int hf_awdl_ht_cap = -1; +static int hf_awdl_ht_ldpc_coding = -1; +static int hf_awdl_ht_chan_width = -1; +static int hf_awdl_ht_sm_pwsave = -1; +static int hf_awdl_ht_green = -1; +static int hf_awdl_ht_short20 = -1; +static int hf_awdl_ht_short40 = -1; +static int hf_awdl_ht_tx_stbc = -1; +static int hf_awdl_ht_rx_stbc = -1; +static int hf_awdl_ht_delayed_block_ack = -1; +static int hf_awdl_ht_max_amsdu = -1; +static int hf_awdl_ht_dss_cck_40 = -1; +static int hf_awdl_ht_psmp = -1; +static int hf_awdl_ht_40_mhz_intolerant = -1; +static int hf_awdl_ht_l_sig = -1; +static int hf_awdl_ampduparam = -1; +static int hf_awdl_ampduparam_mpdu = -1; +static int hf_awdl_ampduparam_mpdu_start_spacing = -1; +static int hf_awdl_ampduparam_reserved = -1; +static int hf_awdl_mcsset = -1; +static int hf_awdl_mcsset_rx_bitmask = -1; +static int hf_awdl_mcsset_rx_bitmask_0to7 = -1; +static int hf_awdl_mcsset_rx_bitmask_8to15 = -1; +static int hf_awdl_mcsset_rx_bitmask_16to23 = -1; +static int hf_awdl_mcsset_rx_bitmask_24to31 = -1; + +static int hf_llc_apple_awdl_pid = -1; + +static gint ett_awdl_data = -1; +static gint ett_awdl = -1; +static gint ett_awdl_fixed_parameters = -1; +static gint ett_awdl_tagged_parameters = -1; +static gint ett_awdl_unknown = -1; +static gint ett_awdl_tag = -1; +static gint ett_awdl_channelseq_flags = -1; +static gint ett_awdl_version = -1; +static gint ett_awdl_dns_record = -1; +static gint ett_awdl_dns_name = -1; +static gint ett_awdl_channelseq_channel_list = -1; +static gint ett_awdl_channelseq_channel = -1; +static gint ett_awdl_datastate_flags = -1; +static gint ett_awdl_datastate_social_channel_map = -1; +static gint ett_awdl_datastate_extflags = -1; +static gint ett_awdl_ht_capabilities = -1; +static gint ett_awdl_ht_ampduparam = -1; +static gint ett_awdl_ht_mcsset_tree = -1; +static gint ett_awdl_ht_mcsbit_tree = -1; +static gint ett_awdl_serviceparams_bitmask = -1; +static gint ett_awdl_serviceparams_values = -1; +static gint ett_awdl_serviceparams_value = -1; + +static expert_field ei_awdl_tag_length = EI_INIT; +static expert_field ei_awdl_tag_data = EI_INIT; +static expert_field ei_awdl_dns_data_len = EI_INIT; + +static dissector_table_t ethertype_subdissector_table; +static dissector_table_t tagged_field_table; + +enum tag_length { + TAG_LENGTH_SHORT = 2, /* short tag header length, used in legacy data frames */ + TAG_LENGTH = 3, /* normal tag header length */ +}; + +enum { + AWDL_SSTH_REQUEST_TLV = 0, + AWDL_SERVICE_REQUEST_TLV = 1, + AWDL_SERVICE_RESPONSE_TLV = 2, + AWDL_UNKNOWN_3_TLV = 3, + AWDL_SYNCHRONIZATON_PARAMETERS_TLV = 4, + AWDL_ELECTION_PARAMETERS_TLV = 5, + AWDL_SERVICE_PARAMETERS_TLV = 6, + AWDL_ENHANCED_DATA_RATE_CAPABILITIES_TLV = 7, + AWDL_ENHANCED_DATA_RATE_OPERATION_TLV = 8, + AWDL_INFRA_TLV = 9, + AWDL_INVITE_TLV = 10, + AWDL_DBG_STRING_TLV = 11, + AWDL_DATA_PATH_STATE_TLV = 12, + AWDL_ENCAPSULATED_IP_TLV = 13, + AWDL_DATAPATH_DEBUG_PACKET_LIVE_TLV = 14, + AWDL_DATAPATH_DEBUG_AF_LIVE_TLV = 15, + AWDL_ARPA_TLV = 16, + AWDL_IEEE80211_CONTAINER_TLV = 17, + AWDL_CHAN_SEQ_TLV = 18, + AWDL_UNKNOWN_19_TLV = 19, + AWDL_SYNCHRONIZATION_TREE_TLV = 20, + AWDL_VERSION_TLV = 21, + AWDL_BLOOM_FILTER_TLV = 22, + AWDL_NAN_SYNC_TLV = 23, + AWDL_ELECTION_PARAMETERS_V2_TLV = 24, +}; + +static const value_string tag_num_vals[] = { + { AWDL_SSTH_REQUEST_TLV, "SSTH Request" }, + { AWDL_SERVICE_REQUEST_TLV, "Service Request" }, + { AWDL_SERVICE_RESPONSE_TLV, "Service Response" }, + { AWDL_UNKNOWN_3_TLV, "Unknown" }, + { AWDL_SYNCHRONIZATON_PARAMETERS_TLV, "Synchronization Parameters" }, + { AWDL_ELECTION_PARAMETERS_TLV, "Election Parameters" }, + { AWDL_SERVICE_PARAMETERS_TLV, "Service Parameters" }, + { AWDL_ENHANCED_DATA_RATE_CAPABILITIES_TLV, "HT Capabilities (IEEE 802.11 subset)" }, + { AWDL_ENHANCED_DATA_RATE_OPERATION_TLV, "Enhanced Data Rate Operation" }, + { AWDL_INFRA_TLV, "Infra" }, + { AWDL_INVITE_TLV, "Invite" }, + { AWDL_DBG_STRING_TLV, "Debug String" }, + { AWDL_DATA_PATH_STATE_TLV, "Data Path State" }, + { AWDL_ENCAPSULATED_IP_TLV, "Encapsulated IP" }, + { AWDL_DATAPATH_DEBUG_PACKET_LIVE_TLV, "Datapath Debug Packet Live" }, + { AWDL_DATAPATH_DEBUG_AF_LIVE_TLV, "Datapath Debug AF Live" }, + { AWDL_ARPA_TLV, "Arpa" }, + { AWDL_IEEE80211_CONTAINER_TLV, "IEEE 802.11 Container" }, + { AWDL_CHAN_SEQ_TLV, "Channel Sequence" }, + { AWDL_UNKNOWN_19_TLV, "Unknown" }, + { AWDL_SYNCHRONIZATION_TREE_TLV, "Synchronization Tree" }, + { AWDL_VERSION_TLV, "Version" }, + { AWDL_BLOOM_FILTER_TLV, "Bloom Filter" }, + { AWDL_NAN_SYNC_TLV, "NAN Sync" }, + { AWDL_ELECTION_PARAMETERS_V2_TLV, "Election Parameters v2" }, + { 0, NULL } +}; +static value_string_ext tag_num_vals_ext = VALUE_STRING_EXT_INIT(tag_num_vals); + +static const value_string awdl_type[] = { + { 8, "AWDL" }, + { 0, NULL } +}; + +enum { + AWDL_SUBTYPE_PSF = 0, + AWDL_SUBTYPE_MIF = 3 +}; + +static const value_string awdl_subtype[] = { + { AWDL_SUBTYPE_PSF, "Periodic Synchronization Frame (PSF)" }, + { AWDL_SUBTYPE_MIF, "Master Indication Frame (MIF)" }, + { 0, NULL } +}; + +static const value_string awdl_subtype_col[] = { + { AWDL_SUBTYPE_PSF, "Periodic Synchronization" }, + { AWDL_SUBTYPE_MIF, "Master Indication" }, + { 0, NULL } +}; + +static const value_string awdl_subtype_short[] = { + { AWDL_SUBTYPE_PSF, "PSF" }, + { AWDL_SUBTYPE_MIF, "MIF" }, + { 0, NULL } +}; + +enum { + AWDL_CHANSEQ_ENC_CHANNELNUMBER = 0, + AWDL_CHANSEQ_ENC_LEGACY = 1, + AWDL_CHANSEQ_ENC_OPCLASS = 3, +}; + +static const value_string awdl_chanseq_enc[] = { + { AWDL_CHANSEQ_ENC_CHANNELNUMBER, "Channelnumber" }, + { AWDL_CHANSEQ_ENC_LEGACY, "Legacy" }, + { AWDL_CHANSEQ_ENC_OPCLASS, "Opclass" }, + { 0, NULL } +}; + +static const value_string awdl_chanseq_control_channel[] = { + { 1, "Lower" }, + { 2, "Upper" }, + { 3, "Primary" }, + { 0, NULL } +}; + +static const value_string awdl_chanseq_bandwidth[] = { + { 1, "20 MHz" }, + { 3, "40 MHz" }, + { 0, NULL } +}; + +static const value_string awdl_chanseq_band[] = { + { 2, "2.4 GHz" }, + { 1, "5 GHz" }, + { 0, NULL } +}; + +static const value_string awdl_chanseq_fill_chan[] = { + { 0xffff, "Repeat Current" }, + { 0, NULL } +}; + +enum { + AWDL_VERSION_MACOS = 1, + AWDL_VERSION_IOS = 2, + AWDL_VERSION_TVOS = 8, +}; + +static const value_string awdl_version_devclass[] = { + { AWDL_VERSION_MACOS, "macOS" }, + { AWDL_VERSION_IOS, "iOS or watchOS" }, + { AWDL_VERSION_TVOS, "tvOS" }, + { 0, NULL } +}; + +enum { + T_PTR = 12, /* domain name pointer */ + T_TXT = 16, /* text strings */ + T_SRV = 33, /* service location (RFC 2052) */ +}; + +static const value_string dns_types_vals[] = { + { T_PTR, "PTR" }, + { T_TXT, "TXT" }, + { T_SRV, "SRV" }, /* RFC 2052 */ + { 0, NULL } +}; + +static const value_string awdl_dns_compression[] = { + { 0xC000, "NULL" }, + { 0xC001, "_airplay._tcp.local" }, + { 0xC002, "_airplay._udp.local" }, + { 0xC003, "_airplay" }, + { 0xC004, "_raop._tcp.local" }, + { 0xC005, "_raop._udp.local" }, + { 0xC006, "_raop" }, + { 0xC007, "_airdrop._tcp.local" }, + { 0xC008, "_airdrop._udp.local" }, + { 0xC009, "_airdrop" }, + { 0xC00A, "_tcp.local" }, + { 0xC00B, "_udp.local" }, + { 0xC00C, "local" }, + { 0xC00D, "ip6.arpa" }, + { 0xC00E, "ip4.arpa" }, + { 0, NULL } +}; + +/* from packet-ieee80211.c */ +static const true_false_string ht_ldpc_coding_flag = { + "Transmitter supports receiving LDPC coded packets", + "Transmitter does not support receiving LDPC coded packets" +}; + +/* from packet-ieee80211.c */ +static const true_false_string ht_chan_width_flag = { + "Transmitter supports 20MHz and 40MHz operation", + "Transmitter only supports 20MHz operation" +}; + +/* from packet-ieee80211.c */ +static const value_string ht_sm_pwsave_flag[] = { + { 0x00, "Static SM Power Save mode" }, + { 0x01, "Dynamic SM Power Save mode" }, + { 0x02, "Reserved" }, + { 0x03, "SM Power Save disabled" }, + { 0x00, NULL} +}; + +/* from packet-ieee80211.c */ +static const true_false_string ht_green_flag = { + "Transmitter is able to receive PPDUs with Green Field (GF) preamble", + "Transmitter is not able to receive PPDUs with Green Field (GF) preamble" +}; + +/* from packet-ieee80211.c */ +static const value_string ht_rx_stbc_flag[] = { + {0x00, "No Rx STBC support"}, + {0x01, "Rx support of one spatial stream"}, + {0x02, "Rx support of one and two spatial streams"}, + {0x03, "Rx support of one, two, and three spatial streams"}, + {0x00, NULL} +}; + +/* from packet-ieee80211.c */ +static const true_false_string ht_delayed_block_ack_flag = { + "Transmitter supports HT-Delayed BlockAck", + "Transmitter does not support HT-Delayed BlockAck" +}; + +/* from packet-ieee80211.c */ +static const true_false_string ht_max_amsdu_flag = { + "7935 bytes", + "3839 bytes" +}; + +/* from packet-ieee80211.c */ +static const true_false_string ht_dss_cck_40_flag = { + "Will/Can use DSSS/CCK in 40 MHz", + "Won't/Can't use of DSSS/CCK in 40 MHz" +}; + +/* from packet-ieee80211.c */ +static const true_false_string ht_psmp_flag = { + "Will/Can support PSMP operation", + "Won't/Can't support PSMP operation" +}; + +/* from packet-ieee80211.c */ +static const true_false_string ht_40_mhz_intolerant_flag = { + "Use of 40 MHz transmissions restricted/disallowed", + "Use of 40 MHz transmissions unrestricted/allowed" +}; + +/* from packet-ieee80211.c */ +static const value_string ampduparam_mpdu_start_spacing_flags[] = { + {0x00, "no restriction"}, + {0x01, "1/4 [usec]"}, + {0x02, "1/2 [usec]"}, + {0x03, "1 [usec]"}, + {0x04, "2 [usec]"}, + {0x05, "4 [usec]"}, + {0x06, "8 [usec]"}, + {0x07, "16 [usec]"}, + {0x00, NULL} +}; + +/* from packet-ieee80211.c */ +static const value_string mcsset_tx_max_spatial_streams_flags[] = { + { 0x00, "1 spatial stream" }, + { 0x01, "2 spatial streams" }, + { 0x02, "3 spatial streams" }, + { 0x03, "4 spatial streams" }, + { 0x04, "TX MCS Set Not Defined" }, + { 0x00, NULL} +}; + +static const value_string apple_awdl_pid_vals[] = { + { 0x0800, "AWDL" }, + { 0, NULL } +}; + +static proto_item * +add_awdl_version(tvbuff_t *tvb, int offset, proto_tree *tree) { + proto_item *version_item; + guint64 version; + static int * const fields[] = { + &hf_awdl_version_major, + &hf_awdl_version_minor, + NULL + }; + + version_item = proto_tree_add_bitmask_with_flags_ret_uint64(tree, tvb, offset, hf_awdl_version, ett_awdl_version, + fields, ENC_LITTLE_ENDIAN, BMT_NO_APPEND, &version); + proto_item_append_text(version_item, " (%u.%u)", (guint8) ((version >> 4) & 0xf), (guint8) (version & 0xf)); + + return version_item; +} + +static int +awdl_tag_version(tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree, void* data _U_) { + int offset = 0; + + add_awdl_version(tvb, offset, tree); + offset += 1; + proto_tree_add_item(tree, hf_awdl_version_devclass, tvb, offset, 1, ENC_LITTLE_ENDIAN); + offset += 1; + + return offset; +} + +static int +awdl_tag_sync_tree(tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree, void* data _U_) { + int tag_len = tvb_reported_length(tvb); + int offset = 0; + + for (; offset <= tag_len - 6; offset += 6) { + proto_tree_add_item(tree, hf_awdl_synctree_addr, tvb, offset, 6, ENC_NA); + } + + return offset; +} + +inline static gboolean +test_bit_guint32(guint i, guint32 n) { + return ((n >> i) & 1) == 1; +} + +static int +awdl_tag_service_params(tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree, void* data _U_) { + proto_item *values_item, *offset_item; + proto_tree *values_tree; + int offset = 0; + + static int * const bitmask_fields[] = { + &hf_awdl_serviceparams_bitmask_0, + &hf_awdl_serviceparams_bitmask_1, + &hf_awdl_serviceparams_bitmask_2, + &hf_awdl_serviceparams_bitmask_3, + &hf_awdl_serviceparams_bitmask_4, + &hf_awdl_serviceparams_bitmask_5, + &hf_awdl_serviceparams_bitmask_6, + &hf_awdl_serviceparams_bitmask_7, + &hf_awdl_serviceparams_bitmask_8, + &hf_awdl_serviceparams_bitmask_9, + &hf_awdl_serviceparams_bitmask_10, + &hf_awdl_serviceparams_bitmask_11, + &hf_awdl_serviceparams_bitmask_12, + &hf_awdl_serviceparams_bitmask_13, + &hf_awdl_serviceparams_bitmask_14, + &hf_awdl_serviceparams_bitmask_15, + &hf_awdl_serviceparams_bitmask_16, + &hf_awdl_serviceparams_bitmask_17, + &hf_awdl_serviceparams_bitmask_18, + &hf_awdl_serviceparams_bitmask_19, + &hf_awdl_serviceparams_bitmask_20, + &hf_awdl_serviceparams_bitmask_21, + &hf_awdl_serviceparams_bitmask_22, + &hf_awdl_serviceparams_bitmask_23, + &hf_awdl_serviceparams_bitmask_24, + &hf_awdl_serviceparams_bitmask_25, + &hf_awdl_serviceparams_bitmask_26, + &hf_awdl_serviceparams_bitmask_27, + &hf_awdl_serviceparams_bitmask_28, + &hf_awdl_serviceparams_bitmask_29, + &hf_awdl_serviceparams_bitmask_30, + &hf_awdl_serviceparams_bitmask_31, + NULL + }; + + static int * const value_fields[] = { + &hf_awdl_serviceparams_values_0, + &hf_awdl_serviceparams_values_1, + &hf_awdl_serviceparams_values_2, + &hf_awdl_serviceparams_values_3, + &hf_awdl_serviceparams_values_4, + &hf_awdl_serviceparams_values_5, + &hf_awdl_serviceparams_values_6, + &hf_awdl_serviceparams_values_7, + NULL + }; + + proto_tree_add_item(tree, hf_awdl_unknown, tvb, offset, 3, ENC_NA); + offset += 3; + proto_tree_add_item(tree, hf_awdl_serviceparams_sui, tvb, offset, 2, ENC_LITTLE_ENDIAN); + offset += 2; + + values_item = proto_tree_add_item(tree, hf_awdl_serviceparams_enc_values, tvb, offset, 0, ENC_NA); /* set length later */ + values_tree = proto_item_add_subtree(values_item, ett_awdl_serviceparams_values); + + offset_item = proto_tree_add_bitmask_with_flags(values_tree, tvb, offset, hf_awdl_serviceparams_bitmask, ett_awdl_serviceparams_bitmask, bitmask_fields, ENC_LITTLE_ENDIAN, BMT_NO_APPEND); + guint32 bitmask = tvb_get_guint32(tvb, offset, ENC_LITTLE_ENDIAN); + offset += 4; + + if (bitmask != 0) { + guint count = 0; + for (guint i = 0; i < 32; i++) { + if (test_bit_guint32(i, bitmask)) { + proto_item *value_item; + guint shift = i << 3; + value_item = proto_tree_add_bitmask(values_tree, tvb, offset, hf_awdl_serviceparams_values, + ett_awdl_serviceparams_value, value_fields, ENC_LITTLE_ENDIAN); + guint8 value = tvb_get_guint8(tvb, offset); + for (guint k = 0; k < 8; k++) { + if (test_bit_guint32(k, value)) { + if (count == 0) { + proto_item_append_text(values_item, ": %u", k + shift); + } else { + proto_item_append_text(values_item, ", %u", k + shift); + } + count++; + } + } + proto_item_append_text(offset_item, ", %u", shift); + proto_item_append_text(value_item, " (offset %u)", shift); + offset++; + } + } + proto_item_set_end(values_item, tvb, offset); + } + + return offset; +} + +static int +awdl_tag_channel_sequence(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void* data _U_) { + proto_item *chanlist_item, *channel_item; + proto_tree *chanlist_tree, *channel_tree; + guint channels; + guint32 chan_number; + wmem_strbuf_t *strbuf; + int offset = 0; + + static int * const flags_fields[] = { + &hf_awdl_channelseq_legacy_control_channel, + &hf_awdl_channelseq_legacy_bandwidth, + &hf_awdl_channelseq_legacy_band, + &hf_awdl_channelseq_legacy_unused, + NULL + }; + + proto_tree_add_item_ret_uint(tree, hf_awdl_channelseq_channel_count, tvb, offset, 1, ENC_LITTLE_ENDIAN, &channels); + channels += 1; /* channel list length is +1 */ + offset += 1; + + guint8 seq_enc = tvb_get_guint8(tvb, offset); + proto_tree_add_item(tree, hf_awdl_channelseq_enc, tvb, offset, 1, ENC_LITTLE_ENDIAN); + offset += 1; + proto_tree_add_item(tree, hf_awdl_channelseq_duplicate, tvb, offset, 1, ENC_LITTLE_ENDIAN); + offset += 1; + proto_tree_add_item(tree, hf_awdl_channelseq_step_count, tvb, offset, 1, ENC_LITTLE_ENDIAN); + offset += 1; + proto_tree_add_item(tree, hf_awdl_channelseq_fill_chan, tvb, offset, 2, ENC_LITTLE_ENDIAN); + offset += 2; + + /* make sufficient space for channel decodings: 5 chars/channel (3-digit number + ', ') */ + strbuf = wmem_strbuf_new_sized(pinfo->pool, 5 * channels); + + switch (seq_enc) { + case AWDL_CHANSEQ_ENC_CHANNELNUMBER: + chanlist_item = proto_tree_add_item(tree, hf_awdl_channelseq_channel_list, tvb, offset, channels, ENC_NA); + chanlist_tree = proto_item_add_subtree(chanlist_item, ett_awdl_channelseq_channel_list); + for (guint i = 0; i < channels; i++) { + proto_tree_add_item_ret_uint(chanlist_tree, hf_awdl_channelseq_channel_number, tvb, offset, 1, ENC_LITTLE_ENDIAN, &chan_number); + offset += 1; + + if (i != 0) { + /* not the first */ + wmem_strbuf_append_printf(strbuf, ", %u", chan_number); + } else { + wmem_strbuf_append_printf(strbuf, "%u", chan_number); + } + } + break; + case AWDL_CHANSEQ_ENC_LEGACY: + chanlist_item = proto_tree_add_item(tree, hf_awdl_channelseq_channel_list, tvb, offset, 2 * channels, ENC_NA); + chanlist_tree = proto_item_add_subtree(chanlist_item, ett_awdl_channelseq_channel_list); + for (guint i = 0; i < channels; i++) { + /* channel number is 2nd byte */ + channel_item = proto_tree_add_item_ret_uint(chanlist_tree, hf_awdl_channelseq_channel, tvb, offset, 2, + ENC_LITTLE_ENDIAN, &chan_number); + channel_tree = proto_item_add_subtree(channel_item, ett_awdl_channelseq_channel); + proto_tree_add_bitmask(channel_tree, tvb, offset, hf_awdl_channelseq_channel_flags, + ett_awdl_channelseq_flags, flags_fields, ENC_LITTLE_ENDIAN); + offset += 1; + proto_tree_add_item(channel_tree, hf_awdl_channelseq_channel_number, tvb, offset, 1, ENC_LITTLE_ENDIAN); + offset += 1; + + if (i != 0) { + /* not the first */ + wmem_strbuf_append_printf(strbuf, ", %u", chan_number); + } else { + wmem_strbuf_append_printf(strbuf, "%u", chan_number); + } + } + break; + case AWDL_CHANSEQ_ENC_OPCLASS: + chanlist_item = proto_tree_add_item(tree, hf_awdl_channelseq_channel_list, tvb, offset, 2 * channels, ENC_NA); + chanlist_tree = proto_item_add_subtree(chanlist_item, ett_awdl_channelseq_channel_list); + for (guint i = 0; i < channels; i++) { + /* channel number is 2nd byte */ + channel_item = proto_tree_add_item_ret_uint(chanlist_tree, hf_awdl_channelseq_channel, tvb, offset, 2, + ENC_LITTLE_ENDIAN, &chan_number); + channel_tree = proto_item_add_subtree(channel_item, ett_awdl_channelseq_channel); + proto_tree_add_item(channel_tree, hf_awdl_channelseq_channel_number, tvb, offset, 1, ENC_LITTLE_ENDIAN); + offset += 1; + proto_tree_add_item(channel_tree, hf_awdl_channelseq_channel_operating_class, tvb, offset, 1, ENC_LITTLE_ENDIAN); + offset += 1; + + if (i != 0) { + /* not the first */ + wmem_strbuf_append_printf(strbuf, ", %u", chan_number); + } else { + wmem_strbuf_append_printf(strbuf, "%u", chan_number); + } + } + break; + default: + /* TODO error handling */ + chanlist_item = NULL; + break; + } + + if (chanlist_item) { + /* finally, append channel list as string */ + proto_item_append_text(chanlist_item, ": %s", wmem_strbuf_get_str(strbuf)); + } + + return offset; +} + +static int +awdl_tag_sync_params(tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree, void* data _U_) { + int tag_len = tvb_reported_length(tvb); + tvbuff_t *chanseq_tvb; + int offset = 0; + + proto_tree_add_item(tree, hf_awdl_syncparams_tx_chan, tvb, offset, 1, ENC_LITTLE_ENDIAN); + offset += 1; + proto_tree_add_item(tree, hf_awdl_syncparams_tx_counter, tvb, offset, 2, ENC_LITTLE_ENDIAN); + offset += 2; + proto_tree_add_item(tree, hf_awdl_syncparams_master_chan, tvb, offset, 1, ENC_LITTLE_ENDIAN); + offset += 1; + proto_tree_add_item(tree, hf_awdl_syncparams_guard_time, tvb, offset, 1, ENC_LITTLE_ENDIAN); + offset += 1; + proto_tree_add_item(tree, hf_awdl_syncparams_aw_period, tvb, offset, 2, ENC_LITTLE_ENDIAN); + offset += 2; + proto_tree_add_item(tree, hf_awdl_syncparams_action_frame_period, tvb, offset, 2, ENC_LITTLE_ENDIAN); + offset += 2; + proto_tree_add_item(tree, hf_awdl_syncparams_awdl_flags, tvb, offset, 2, ENC_LITTLE_ENDIAN); + offset += 2; + proto_tree_add_item(tree, hf_awdl_syncparams_aw_ext_length, tvb, offset, 2, ENC_LITTLE_ENDIAN); + offset += 2; + proto_tree_add_item(tree, hf_awdl_syncparams_aw_cmn_length, tvb, offset, 2, ENC_LITTLE_ENDIAN); + offset += 2; + proto_tree_add_item(tree, hf_awdl_syncparams_aw_remaining, tvb, offset, 2, ENC_LITTLE_ENDIAN); + offset += 2; + proto_tree_add_item(tree, hf_awdl_syncparams_ext_min, tvb, offset, 1, ENC_LITTLE_ENDIAN); + offset += 1; + proto_tree_add_item(tree, hf_awdl_syncparams_ext_max_multi, tvb, offset, 1, ENC_LITTLE_ENDIAN); + offset += 1; + proto_tree_add_item(tree, hf_awdl_syncparams_ext_max_uni, tvb, offset, 1, ENC_LITTLE_ENDIAN); + offset += 1; + proto_tree_add_item(tree, hf_awdl_syncparams_ext_max_af, tvb, offset, 1, ENC_LITTLE_ENDIAN); + offset += 1; + + //the following values are also used in IOFamily + proto_tree_add_item(tree, hf_awdl_syncparams_master, tvb, offset, 6, ENC_NA); + offset += 6; + proto_tree_add_item(tree, hf_awdl_syncparams_presence_mode, tvb, offset, 1, ENC_NA); + offset += 1; + proto_tree_add_item(tree, hf_awdl_unknown, tvb, offset, 1, ENC_NA); + offset += 1; + proto_tree_add_item(tree, hf_awdl_syncparams_awcounter, tvb, offset, 2, ENC_LITTLE_ENDIAN); + offset += 2; + proto_tree_add_item(tree, hf_awdl_syncparams_apbeaconalignment, tvb, offset, 2, ENC_LITTLE_ENDIAN); + offset += 2; + + chanseq_tvb = tvb_new_subset_length(tvb, offset, tag_len - offset); + offset += awdl_tag_channel_sequence(chanseq_tvb, pinfo, tree, data); + + return offset; +} + +static int +awdl_tag_election_params(tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree, void* data _U_) { + int offset = 0; + + guint8 private_election = tvb_get_guint8(tvb, offset); + + proto_tree_add_item(tree, hf_awdl_electionparams_flags, tvb, offset, 1, ENC_NA); + offset += 1; + proto_tree_add_item(tree, hf_awdl_electionparams_id, tvb, offset, 2, ENC_NA); + offset += 2; + proto_tree_add_item(tree, hf_awdl_electionparams_distance, tvb, offset, 1, ENC_NA); + offset += 1; + proto_tree_add_item(tree, hf_awdl_electionparams_unknown, tvb, offset, 1, ENC_NA); + offset += 1; + proto_tree_add_item(tree, hf_awdl_electionparams_master, tvb, offset, 6, ENC_NA); + offset += 6; + proto_tree_add_item(tree, hf_awdl_electionparams_mastermetric, tvb, offset, 4, ENC_LITTLE_ENDIAN); + offset += 4; + proto_tree_add_item(tree, hf_awdl_electionparams_selfmetric, tvb, offset, 4, ENC_LITTLE_ENDIAN); + offset += 4; + + if (private_election) { + proto_tree_add_item(tree, hf_awdl_unknown, tvb, offset, 2, ENC_NA); + offset += 2; + proto_tree_add_item(tree, hf_awdl_electionparams_private_master, tvb, offset, 6, ENC_NA); + offset += 6; + proto_tree_add_item(tree, hf_awdl_electionparams_private_mastermetric, tvb, offset, 4, ENC_LITTLE_ENDIAN); + offset += 4; + proto_tree_add_item(tree, hf_awdl_electionparams_private_id, tvb, offset, 4, ENC_LITTLE_ENDIAN); + offset += 4; + proto_tree_add_item(tree, hf_awdl_electionparams_private_phc, tvb, offset, 2, ENC_LITTLE_ENDIAN); + offset += 2; + } + + return offset; +} + +static int +awdl_tag_election_params_v2(tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree, void* data _U_) { + int offset = 0; + + proto_tree_add_item(tree, hf_awdl_electionparams2_master, tvb, offset, 6, ENC_NA); + offset += 6; + proto_tree_add_item(tree, hf_awdl_electionparams2_other, tvb, offset, 6, ENC_NA); + offset += 6; + proto_tree_add_item(tree, hf_awdl_electionparams2_mastercounter, tvb, offset, 4, ENC_LITTLE_ENDIAN); + offset += 4; + proto_tree_add_item(tree, hf_awdl_electionparams2_distance, tvb, offset, 4, ENC_LITTLE_ENDIAN); + offset += 4; + proto_tree_add_item(tree, hf_awdl_electionparams2_mastermetric, tvb, offset, 4, ENC_LITTLE_ENDIAN); + offset += 4; + proto_tree_add_item(tree, hf_awdl_electionparams2_selfmetric, tvb, offset, 4, ENC_LITTLE_ENDIAN); + offset += 4; + proto_tree_add_item(tree, hf_awdl_electionparams2_unknown, tvb, offset, 4, ENC_LITTLE_ENDIAN); + offset += 4; + proto_tree_add_item(tree, hf_awdl_electionparams2_reserved, tvb, offset, 4, ENC_LITTLE_ENDIAN); + offset += 4; + proto_tree_add_item(tree, hf_awdl_electionparams2_selfcounter, tvb, offset, 4, ENC_LITTLE_ENDIAN); + offset += 4; + + return offset; +} + +static int +awdl_tag_datapath_state(tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree, void* data _U_) { + int offset = 0; + guint16 flags; + + static int * const flags_fields[] = { + &hf_awdl_datastate_flags_0, + &hf_awdl_datastate_flags_1, + &hf_awdl_datastate_flags_2, + &hf_awdl_datastate_flags_3, + &hf_awdl_datastate_flags_4, + &hf_awdl_datastate_flags_5, + &hf_awdl_datastate_flags_6, + &hf_awdl_datastate_flags_7, + &hf_awdl_datastate_flags_8, + &hf_awdl_datastate_flags_9, + &hf_awdl_datastate_flags_10, + &hf_awdl_datastate_flags_11, + &hf_awdl_datastate_flags_12, + &hf_awdl_datastate_flags_13, + &hf_awdl_datastate_flags_14, + &hf_awdl_datastate_flags_15, + NULL + }; + + static int * const channel_map_fields[] = { + &hf_awdl_datastate_social_channel_map_6, + &hf_awdl_datastate_social_channel_map_44, + &hf_awdl_datastate_social_channel_map_149, + &hf_awdl_datastate_social_channel_map_unused, + NULL + }; + + static int * const extflags_fields[] = { + &hf_awdl_datastate_extflags_0, + &hf_awdl_datastate_extflags_1, + &hf_awdl_datastate_extflags_2, + &hf_awdl_datastate_extflags_3to15, + NULL + }; + + flags = tvb_get_guint16(tvb, offset, ENC_LITTLE_ENDIAN); + proto_tree_add_bitmask(tree, tvb, offset, hf_awdl_datastate_flags, + ett_awdl_datastate_flags, flags_fields, ENC_LITTLE_ENDIAN); + offset += 2; + + if (flags & 0x0100) { + proto_tree_add_item(tree, hf_awdl_datastate_countrycode, tvb, offset, 3, ENC_ASCII); + offset += 3; + } + if (flags & 0x0200) { + /* this can either be a channel or a map indicating which channels this node supports */ + guint16 map = tvb_get_guint16(tvb, offset, ENC_LITTLE_ENDIAN); + /* TODO unverified heuristic to decide whether this is a map or number */ + if (map & 1) { + proto_tree_add_bitmask(tree, tvb, offset, hf_awdl_datastate_social_channel_map, + ett_awdl_datastate_social_channel_map, channel_map_fields, ENC_LITTLE_ENDIAN); + } else { + /* a single channel number */ + proto_tree_add_item(tree, hf_awdl_datastate_social_channel, tvb, offset, 2, ENC_LITTLE_ENDIAN); + } + offset += 2; + } + if (flags & 0x0001) { + proto_tree_add_item(tree, hf_awdl_datastate_infra_bssid, tvb, offset, 6, ENC_NA); + proto_tree_add_item(tree, hf_awdl_datastate_infra_channel, tvb, offset + 6, 2, ENC_LITTLE_ENDIAN); + offset += 8; + } + if (flags & 0x0002) { + // if not set, this will be the same as 0x1 + proto_tree_add_item(tree, hf_awdl_datastate_infra_address, tvb, offset, 6, ENC_NA); + offset += 6; + } + if (flags & 0x0004) { + proto_tree_add_item(tree, hf_awdl_datastate_awdl_address, tvb, offset, 6, ENC_NA); + offset += 6; + } + if (flags & 0x0010) { + proto_tree_add_item(tree, hf_awdl_datastate_umi, tvb, offset, 2, ENC_LITTLE_ENDIAN); + offset += 2; + } + if (flags & 0x1000) { + guint16 optionlength = tvb_get_guint16(tvb, offset, ENC_LITTLE_ENDIAN); + proto_tree_add_item(tree, hf_awdl_datastate_umioptions_length, tvb, offset, 2, ENC_LITTLE_ENDIAN); + offset += 2; + proto_tree_add_item(tree, hf_awdl_datastate_umioptions, tvb, offset, optionlength, ENC_NA); + offset += optionlength; + } + /* now come the extended parameters */ + if (flags & 0x8000) { + guint16 extflags = tvb_get_guint16(tvb, offset, ENC_LITTLE_ENDIAN); + proto_tree_add_bitmask(tree, tvb, offset, hf_awdl_datastate_extflags, + ett_awdl_datastate_extflags, extflags_fields, ENC_LITTLE_ENDIAN); + offset += 2; + + if (extflags & 0x4) { + proto_tree_add_item(tree, hf_awdl_datastate_logtrigger, tvb, offset, 4, ENC_LITTLE_ENDIAN); + offset += 4; + } + if (extflags & 0x1) { + /* TODO add actual fields + * + * Binary: IO80211Family + * Method: IO80211AWDLPeer::parseAwdlDataPathTLVAndTakeAction + * + * __cstring:0000000000091E2A aRlfc db 'RLFC',0 + * __cstring:0000000000091E2F aMsecsince db 'msecSince',0 + * __cstring:0000000000091E39 aAwseq db 'awSeq',0 + * __cstring:0000000000091E3F aPayupdatecount db 'payUpdateCounter',0 + */ + proto_tree_add_item(tree, hf_awdl_datastate_undecoded, tvb, offset, 14, ENC_NA); + offset += 14; + } + } + + return offset; +} + +static int +awdl_tag_ieee80211_container(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void* data _U_) { + int offset = 0; + + const guint8 ids[] = { + 191, // VHT Capability + }; + offset += add_tagged_field(pinfo, tree, tvb, offset, MGT_ACTION, ids, G_N_ELEMENTS(ids), NULL); + + return offset; +} + +static int +awdl_tag_ht_capabilities(tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree, void* data _U_) { + proto_item *ti, *cap_item; + proto_tree *mcs_tree, *bit_tree, *cap_tree; + guint8 streams; /* 0-4 for HT and 0-8 for VHT*/ + int offset = 0; + int tag_len = tvb_reported_length(tvb); + + static int * const awdl_ht[] = { + &hf_awdl_ht_ldpc_coding, + &hf_awdl_ht_chan_width, + &hf_awdl_ht_sm_pwsave, + &hf_awdl_ht_green, + &hf_awdl_ht_short20, + &hf_awdl_ht_short40, + &hf_awdl_ht_tx_stbc, + &hf_awdl_ht_rx_stbc, + &hf_awdl_ht_delayed_block_ack, + &hf_awdl_ht_max_amsdu, + &hf_awdl_ht_dss_cck_40, + &hf_awdl_ht_psmp, + &hf_awdl_ht_40_mhz_intolerant, + &hf_awdl_ht_l_sig, + NULL + }; + + proto_tree_add_item(tree, hf_awdl_ht_unknown, tvb, offset, 2, ENC_LITTLE_ENDIAN); + offset += 2; + + proto_tree_add_bitmask_with_flags(tree, tvb, offset, hf_awdl_ht_cap, ett_awdl_ht_capabilities, + awdl_ht, ENC_LITTLE_ENDIAN, BMT_NO_APPEND); + offset += 2; + + cap_item = proto_tree_add_item(tree, hf_awdl_ampduparam, tvb, offset, 1, ENC_LITTLE_ENDIAN); + cap_tree = proto_item_add_subtree(cap_item, ett_awdl_ht_ampduparam); + ti = proto_tree_add_item(cap_tree, hf_awdl_ampduparam_mpdu, tvb, offset, 1, ENC_LITTLE_ENDIAN); + proto_item_append_text(ti, " (%04.0f[Bytes])", pow(2, 13 + (tvb_get_guint8(tvb, offset) & 0x3)) - 1); + proto_tree_add_item(cap_tree, hf_awdl_ampduparam_mpdu_start_spacing, tvb, offset, 1, ENC_LITTLE_ENDIAN); + proto_tree_add_item(cap_tree, hf_awdl_ampduparam_reserved, tvb, offset, 1, ENC_LITTLE_ENDIAN); + offset += 1; + + /* Check how many streams are supported */ + for (streams = 0; streams < 4 /* max streams */ && tvb_get_guint8(tvb, offset + streams) != 0; streams++) { + } + + ti = proto_tree_add_item(tree, hf_awdl_mcsset, tvb, offset, streams, ENC_NA); + mcs_tree = proto_item_add_subtree(ti, ett_awdl_ht_mcsset_tree); + + /* Rx MCS Bitmask */ + ti = proto_tree_add_item(mcs_tree, hf_awdl_mcsset_rx_bitmask, tvb, offset, streams, ENC_NA); + bit_tree = proto_item_add_subtree(ti, ett_awdl_ht_mcsbit_tree); + + proto_tree_add_item(bit_tree, hf_awdl_mcsset_rx_bitmask_0to7, tvb, offset, streams, ENC_LITTLE_ENDIAN); + offset += 1; + if (offset < tag_len - 2) { + proto_tree_add_item(bit_tree, hf_awdl_mcsset_rx_bitmask_8to15, tvb, offset - 1, streams, ENC_LITTLE_ENDIAN); + offset += 1; + } + if (offset < tag_len - 2) { + proto_tree_add_item(bit_tree, hf_awdl_mcsset_rx_bitmask_16to23, tvb, offset - 2, streams, ENC_LITTLE_ENDIAN); + offset += 1; + } + if (offset < tag_len - 2) { + proto_tree_add_item(bit_tree, hf_awdl_mcsset_rx_bitmask_24to31, tvb, offset - 3, streams, ENC_LITTLE_ENDIAN); + offset += 1; + } + + proto_item_append_text(ti, ": %s", val_to_str(streams - 1, mcsset_tx_max_spatial_streams_flags, "Reserved: %d" ) ); + + // Some padding at the end + proto_tree_add_item(tree, hf_awdl_ht_unknown, tvb, offset, 2, ENC_LITTLE_ENDIAN); + offset += 2; + + return offset; +} + +/* + * Decodes a AWDL-variant DNS name. + * + * 'hfindex_regular' needs to registered as + * 'hfindex_compressed' assumes a field that can be decoded with the 'awdl_dns_compression' value_string + */ +static int +add_awdl_dns_name(proto_tree *tree, int hfindex_regular, int hfindex_compressed, + tvbuff_t *tvb, int offset, int len, wmem_allocator_t *scope, const gchar **name) { + int start_offset = offset; + guint8 component_len; + const guchar *component; + wmem_strbuf_t *strbuf; + + strbuf = wmem_strbuf_new_sized(scope, MAX_DNAME_LEN); + + while (offset < (len + start_offset)) { + component_len = tvb_get_guint8(tvb, offset); + if (component_len & 0xC0) { + /* compressed label */ + guint compressed_value; + proto_tree_add_item_ret_uint(tree, hfindex_compressed, tvb, offset, 2, ENC_BIG_ENDIAN, &compressed_value); + if (compressed_value == 0xC000) { + // 'NULL' compression -> ignore in printed string + component = NULL; + } else { + component = val_to_str_const(compressed_value, awdl_dns_compression, "<UNKNOWN>"); + } + offset += 2; + } else { + /* regular label */ + guint label_len; + proto_tree_add_item_ret_string_and_length(tree, hfindex_regular, tvb, offset, 1, ENC_ASCII, scope, &component, &label_len); + offset += label_len; + } + if (component) { + if (wmem_strbuf_get_len(strbuf)) + /* not the first entry */ + wmem_strbuf_append_c(strbuf, '.'); + wmem_strbuf_append(strbuf, component); + } + } + + *name = wmem_strbuf_get_str(strbuf); + + return offset - start_offset; +} + +static int +add_awdl_dns_entry(packet_info *pinfo, proto_tree *tree, gint ett, + int hfindex_entry, int hfindex_regular, int hfindex_compressed, + tvbuff_t *tvb, int offset, int len, const gchar **name) { + int start_offset = offset; + proto_item *entry_item; + proto_tree *entry_tree; + const gchar *n; + + entry_item = proto_tree_add_item(tree, hfindex_entry, tvb, offset, 0, ENC_NA); + entry_tree = proto_item_add_subtree(entry_item, ett); + offset += add_awdl_dns_name(entry_tree, hfindex_regular, hfindex_compressed, tvb, offset, len, pinfo->pool, &n); + proto_item_set_end(entry_item, tvb, offset); + proto_item_append_text(entry_item, ": %s", n); + + if (name) + (*name) = n; + + return offset - start_offset; +} + +static int +awdl_tag_arpa(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void* data _U_) { + int offset = 0; + int tag_len = tvb_reported_length(tvb); + + proto_tree_add_item(tree, hf_awdl_arpa_flags, tvb, offset, 1, ENC_NA); + offset += 1; + offset += add_awdl_dns_entry(pinfo, tree, ett_awdl_dns_name, hf_awdl_arpa, hf_awdl_arpa_name, + hf_awdl_arpa_short, tvb, offset, tag_len - offset, NULL); + + return offset; +} + +static int +awdl_tag_service_response(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void* data _U_) { + proto_item *rr_item; + proto_tree *rr_tree, *data_len; + const gchar *name; + int offset = 0; + guint len, type; + guint prio, weight, port; + + rr_tree = proto_tree_add_subtree(tree, tvb, offset, -1, ett_awdl_dns_record, &rr_item, ""); + + proto_tree_add_item_ret_uint(rr_tree, hf_awdl_dns_name_len, tvb, offset, 2, ENC_LITTLE_ENDIAN, &len); + offset += 2; + + // len field includes the following type value + len -= 1; + offset += add_awdl_dns_entry(pinfo, rr_tree, ett_awdl_dns_name, hf_awdl_dns_name, hf_awdl_dns_name_label, + hf_awdl_dns_name_short, tvb, offset, len, &name); + + proto_tree_add_item_ret_uint(rr_tree, hf_awdl_dns_type, tvb, offset, 1, ENC_LITTLE_ENDIAN, &type); + offset += 1; + + proto_item_set_text(rr_item, "%s: type %s", name, val_to_str_const(type, dns_types_vals, "UNKNOWN")); + + data_len = proto_tree_add_item_ret_uint(rr_tree, hf_awdl_dns_data_len, tvb, offset, 2, ENC_LITTLE_ENDIAN, &len); + offset += 2; + // TODO could be that len field is actually uint32? + proto_tree_add_item(rr_tree, hf_awdl_dns_unknown, tvb, offset, 2, ENC_LITTLE_ENDIAN); + offset += 2; + + switch (type) { + case T_TXT: + while (len > 0) { + const guchar *txt; + gint label_len; + proto_tree_add_item_ret_string_and_length(rr_tree, hf_awdl_dns_txt, tvb, offset, 1, ENC_ASCII, + pinfo->pool, &txt, &label_len); + offset += label_len; + proto_item_append_text(rr_item, ", %s", txt); + if (label_len > (gint) len) { + expert_add_info_format(pinfo, data_len, &ei_awdl_tag_length, + "DNS data length is too short"); + break; + } + len -= label_len; + } + break; + case T_SRV: + proto_tree_add_item_ret_uint(rr_tree, hf_awdl_dns_priority, tvb, offset, 2, ENC_BIG_ENDIAN, &prio); + offset += 2; + proto_tree_add_item_ret_uint(rr_tree, hf_awdl_dns_weight, tvb, offset, 2, ENC_BIG_ENDIAN, &weight); + offset += 2; + proto_tree_add_item_ret_uint(rr_tree, hf_awdl_dns_port, tvb, offset, 2, ENC_BIG_ENDIAN, &port); + offset += 2; + // length field includes above fields + len -= 6; + offset += add_awdl_dns_entry(pinfo, rr_tree, ett_awdl_dns_name, hf_awdl_dns_target, hf_awdl_dns_target_label, + hf_awdl_dns_target_short, tvb, offset, len, &name); + proto_item_append_text(rr_item, ", priority %u, weight %u, port %u, target %s", prio, weight, port, name); + break; + case T_PTR: + offset += add_awdl_dns_entry(pinfo, rr_tree, ett_awdl_dns_name, hf_awdl_dns_ptr, hf_awdl_dns_ptr_label, + hf_awdl_dns_ptr_short, tvb, offset, len, &name); + proto_item_append_text(rr_item, ", %s", name); + break; + default: + break; + } + + return offset; +} + +static int +awdl_add_tagged_field(packet_info *pinfo, proto_tree *tree, tvbuff_t *tvb, int offset, enum tag_length hdr_len) { + tvbuff_t *tag_tvb; + guint32 tag_no, tag_len; + proto_tree *orig_tree = tree; + proto_item *ti = NULL; + proto_item *ti_len, *ti_tag; + awdl_tagged_field_data_t field_data; + int parsed; + + tag_no = tvb_get_guint8(tvb, offset); + if (hdr_len == TAG_LENGTH_SHORT) { + tag_len = tvb_get_guint8(tvb, offset + 1); + } else { + tag_len = tvb_get_guint16(tvb, offset + 1, ENC_LITTLE_ENDIAN); + } + + if (tree) { + ti = proto_tree_add_item(orig_tree, hf_awdl_tag, tvb, offset, tag_len + hdr_len, ENC_NA); + proto_item_append_text(ti, ": %s", val_to_str_ext(tag_no, &tag_num_vals_ext, "Unknown (%d)")); + tree = proto_item_add_subtree(ti, ett_awdl_tag); + } + + ti_tag = proto_tree_add_uint(tree, hf_awdl_tag_number, tvb, offset, 1, tag_no); + ti_len = proto_tree_add_uint(tree, hf_awdl_tag_length, tvb, offset + 1, hdr_len - 1, tag_len); + offset += hdr_len; + if (tag_len > (guint)tvb_reported_length_remaining(tvb, offset)) { + expert_add_info_format(pinfo, ti_len, &ei_awdl_tag_length, + "Tag Length is longer than remaining payload"); + } + + tag_tvb = tvb_new_subset_length(tvb, offset, tag_len); + field_data.item_tag = ti; + field_data.item_tag_length = ti_len; + if (!(parsed = dissector_try_uint_new(tagged_field_table, tag_no, tag_tvb, pinfo, tree, FALSE, &field_data))) + { + proto_tree_add_item(tree, hf_awdl_tag_data, tag_tvb, 0, tag_len, ENC_NA); + expert_add_info_format(pinfo, ti_tag, &ei_awdl_tag_data, + "Dissector for AWDL tag (%s) code not implemented", + val_to_str_ext(tag_no, &tag_num_vals_ext, "(%d)")); + proto_item_append_text(ti, ": Undecoded"); + } + else if (parsed > 0 && (unsigned int) parsed < tag_len) + { + proto_tree_add_item(tree, hf_awdl_tag_padding, tag_tvb, parsed, tag_len - parsed, ENC_NA); + } + + return tag_len + hdr_len; +} + +static void +awdl_add_tagged_parameters(tvbuff_t *tvb, int offset, packet_info *pinfo, proto_tree *tree, int tagged_parameters_len) +{ + int next_len; + while (tagged_parameters_len > 0) { + if ((next_len = awdl_add_tagged_field(pinfo, tree, tvb, offset, TAG_LENGTH)) == 0) + break; + if (next_len > tagged_parameters_len) { + /* XXX - flag this as an error? */ + next_len = tagged_parameters_len; + } + offset += next_len; + tagged_parameters_len -= next_len; + } +} + +static proto_tree * +get_tagged_parameter_tree(proto_tree * tree, tvbuff_t *tvb, int start, int size) +{ + proto_item *tagged_fields; + + tagged_fields = proto_tree_add_item(tree, hf_awdl_tagged_parameters, tvb, start, -1, ENC_NA); + proto_item_append_text(tagged_fields, " (%d bytes)", size); + + return proto_item_add_subtree(tagged_fields, ett_awdl_tagged_parameters); +} + +static int +dissect_awdl_action(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void *data _U_) +{ + int offset = 0; + gint tagged_parameter_tree_len; + proto_tree *parent, *af_tree, *fixed_tree, *tag_tree; + proto_item *ti, *item, *fixed_fields; + guint32 phytime, targettime; + guint8 subtype; + + parent = proto_tree_get_parent_tree(proto_tree_get_parent_tree(tree)); + ti = proto_tree_add_item(parent, proto_awdl, tvb, offset, -1, ENC_NA); + af_tree = proto_item_add_subtree(ti, ett_awdl); + + fixed_fields = proto_tree_add_item(af_tree, hf_awdl_fixed_parameters, tvb, offset, 12, ENC_NA); + fixed_tree = proto_item_add_subtree(fixed_fields, ett_awdl_fixed_parameters); + proto_tree_add_item(fixed_tree, hf_awdl_type, tvb, offset, 1, ENC_LITTLE_ENDIAN); + offset += 1; + add_awdl_version(tvb, offset, fixed_tree); + offset += 1; + subtype = tvb_get_guint8(tvb, offset); + proto_tree_add_item(fixed_tree, hf_awdl_subtype, tvb, offset, 1, ENC_LITTLE_ENDIAN); + offset += 1; + proto_tree_add_item(fixed_tree, hf_awdl_rsvd, tvb, offset, 1, ENC_LITTLE_ENDIAN); + offset += 1; + proto_tree_add_item(fixed_tree, hf_awdl_phytime, tvb, offset, 4, ENC_LITTLE_ENDIAN); + phytime = tvb_get_guint32(tvb, offset, ENC_LITTLE_ENDIAN); + offset += 4; + proto_tree_add_item(fixed_tree, hf_awdl_targettime, tvb, offset, 4, ENC_LITTLE_ENDIAN); + targettime = tvb_get_guint32(tvb, offset, ENC_LITTLE_ENDIAN); + offset += 4; + item = proto_tree_add_uint(fixed_tree, hf_awdl_txdelay, tvb, 0, 0, phytime - targettime); + proto_item_set_generated(item); + + col_set_str(pinfo->cinfo, COL_PROTOCOL, "AWDL"); + col_set_str(pinfo->cinfo, COL_INFO, val_to_str_const(subtype, awdl_subtype_col, "Unknown")); + proto_item_append_text(ti, ", Subtype: %s", val_to_str_const(subtype, awdl_subtype_short, "Unknown")); + + tagged_parameter_tree_len = tvb_reported_length_remaining(tvb, offset); + tag_tree = get_tagged_parameter_tree(af_tree, tvb, offset, tagged_parameter_tree_len); + awdl_add_tagged_parameters(tvb, offset, pinfo, tag_tree, tagged_parameter_tree_len); + offset += tagged_parameter_tree_len; + + return offset; +} + +static int +dissect_awdl_data(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void* data _U_) +{ + gint offset = 0; + guint etype; + tvbuff_t *next_tvb; + proto_item *ti; + proto_tree *awdl_tree; + guint seq; + + col_set_str(pinfo->cinfo, COL_PROTOCOL, "AWDL Data"); + col_clear(pinfo->cinfo, COL_INFO); + + ti = proto_tree_add_item(tree, proto_awdl_data, tvb, 0, -1, ENC_NA); + + awdl_tree = proto_item_add_subtree(ti, ett_awdl_data); + + proto_tree_add_item(awdl_tree, hf_awdl_data_header, tvb, offset, 2, ENC_NA); + offset += 2; + + proto_tree_add_item_ret_uint(awdl_tree, hf_awdl_data_seq, tvb, offset, 2, ENC_LITTLE_ENDIAN, &seq); + col_add_fstr(pinfo->cinfo, COL_INFO, "Seq=%u", seq); + proto_item_append_text(ti, ", Seq: %u", seq); + offset += 2; + + if (tvb_get_guint8(tvb, offset) == 3) { + // 0x0300 ("long format") + proto_item *tagged_item; + proto_tree *tagged_tree; + int start_offset; + guint8 slen; + + slen = tvb_get_guint8(tvb, offset + 1); + proto_tree_add_item(awdl_tree, hf_awdl_data_header, tvb, offset, 2 + slen, ENC_NA); + offset += 2 + slen; + + tagged_item = proto_tree_add_item(awdl_tree, hf_awdl_tagged_parameters, tvb, offset, 0, ENC_NA); /* set length later */ + tagged_tree = proto_item_add_subtree(tagged_item, ett_awdl_tagged_parameters); + + start_offset = offset; + + while (tvb_get_guint8(tvb, offset) != 3) { + offset += awdl_add_tagged_field(pinfo, tagged_tree, tvb, offset, TAG_LENGTH_SHORT); + } + + slen = tvb_get_guint8(tvb, offset + 1); + proto_tree_add_item(awdl_tree, hf_awdl_data_header, tvb, offset, 2 + slen, ENC_NA); + offset += 2 + slen; + + proto_item_set_len(tagged_item, offset - start_offset); + proto_item_append_text(tagged_item, " (%d bytes)", offset - start_offset); + + } + else { + // 0x0000 + // TODO: should have some sanity check. + proto_tree_add_item(awdl_tree, hf_awdl_data_header, tvb, offset, 2, ENC_NA); + offset += 2; + } + + /* Last is some ethertype */ + proto_tree_add_item_ret_uint(awdl_tree, hf_awdl_data_ethertype, tvb, offset, 2, ENC_BIG_ENDIAN, &etype); + offset += 2; + + proto_item_set_len(awdl_tree, offset); + + next_tvb = tvb_new_subset_remaining(tvb, offset); + if (!dissector_try_uint(ethertype_subdissector_table, etype, next_tvb, pinfo, tree)) + call_data_dissector(next_tvb, pinfo, tree); + + return tvb_captured_length(tvb); +} + +static void +awdl_register_tags(void) +{ + dissector_add_uint("awdl.tag.number", AWDL_SERVICE_RESPONSE_TLV, create_dissector_handle(awdl_tag_service_response, proto_awdl)); + dissector_add_uint("awdl.tag.number", AWDL_SYNCHRONIZATON_PARAMETERS_TLV, create_dissector_handle(awdl_tag_sync_params, proto_awdl)); + dissector_add_uint("awdl.tag.number", AWDL_ELECTION_PARAMETERS_TLV, create_dissector_handle(awdl_tag_election_params, proto_awdl)); + dissector_add_uint("awdl.tag.number", AWDL_SERVICE_PARAMETERS_TLV, create_dissector_handle(awdl_tag_service_params, proto_awdl)); + dissector_add_uint("awdl.tag.number", AWDL_ENHANCED_DATA_RATE_CAPABILITIES_TLV, create_dissector_handle(awdl_tag_ht_capabilities, proto_awdl)); + dissector_add_uint("awdl.tag.number", AWDL_DATA_PATH_STATE_TLV, create_dissector_handle(awdl_tag_datapath_state, proto_awdl)); + dissector_add_uint("awdl.tag.number", AWDL_ARPA_TLV, create_dissector_handle(awdl_tag_arpa, proto_awdl)); + dissector_add_uint("awdl.tag.number", AWDL_IEEE80211_CONTAINER_TLV, create_dissector_handle(awdl_tag_ieee80211_container, proto_awdl)); + dissector_add_uint("awdl.tag.number", AWDL_CHAN_SEQ_TLV, create_dissector_handle(awdl_tag_channel_sequence, proto_awdl)); + dissector_add_uint("awdl.tag.number", AWDL_SYNCHRONIZATION_TREE_TLV, create_dissector_handle(awdl_tag_sync_tree, proto_awdl)); + dissector_add_uint("awdl.tag.number", AWDL_VERSION_TLV, create_dissector_handle(awdl_tag_version, proto_awdl)); + dissector_add_uint("awdl.tag.number", AWDL_ELECTION_PARAMETERS_V2_TLV, create_dissector_handle(awdl_tag_election_params_v2, proto_awdl)); +} + +void proto_register_awdl(void) +{ + static hf_register_info hf[] = { + /* Default for unknown fields */ + { &hf_awdl_unknown, + { "Unknown", "awdl.unknown", + FT_BYTES, BASE_NONE, NULL, 0x0, NULL, HFILL + } + }, + + /* LLC */ + { &hf_awdl_data_seq, + { "Sequence number", "awdl_data.seq", + FT_UINT16, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_data_header, + { "Header data", "awdl_data.header", + FT_BYTES, BASE_NONE, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_data_ethertype, + { "EtherType", "awdl_data.ethertype", + FT_UINT16, BASE_HEX, NULL, 0x0, NULL, HFILL + } + }, + + /* Action Frame */ + { &hf_awdl_fixed_parameters, + { "Fixed parameters", "awdl.fixed.all", + FT_NONE, BASE_NONE, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_tagged_parameters, + { "Tagged parameters", "awdl.tagged.all", + FT_NONE, BASE_NONE, NULL, 0, NULL, HFILL + } + }, + + /* Fixed parameters */ + { &hf_awdl_type, + { "Type", "awdl.type", + FT_UINT8, BASE_DEC, VALS(awdl_type), 0x0, NULL, HFILL + } + }, + { &hf_awdl_subtype, + { "Subtype", "awdl.subtype", + FT_UINT8, BASE_DEC, VALS(awdl_subtype), 0x0, NULL, HFILL + } + }, + { &hf_awdl_rsvd, + { "Reserved", "awdl.reserved", + FT_UINT8, BASE_HEX, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_phytime, + { "PHY Tx Time", "awdl.phytime", + FT_UINT32, BASE_DEC | BASE_UNIT_STRING, &units_microseconds, 0x0, + "Time shortly before the frame was sent out by the radio", HFILL + } + }, + { &hf_awdl_targettime, + { "Target Tx Time", "awdl.targettime", + FT_UINT32, BASE_DEC | BASE_UNIT_STRING, &units_microseconds, 0x0, + "Time when the frame was created.", HFILL + } + }, + { &hf_awdl_txdelay, + { "Tx Delay", "awdl.txdelay", + FT_UINT32, BASE_DEC | BASE_UNIT_STRING, &units_microseconds, 0x0, + "Difference between the PHY and target time stamps", HFILL + } + }, + + /* TLV */ + { &hf_awdl_tag, + { "Tag", "awdl.tag", + FT_NONE, BASE_NONE, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_tag_number, + { "Tag Number", "awdl.tag.number", + FT_UINT8, BASE_DEC | BASE_EXT_STRING, &tag_num_vals_ext, 0x0, NULL, HFILL + } + }, + { &hf_awdl_tag_length, + { "Tag Length", "awdl.tag.length", + FT_UINT16, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_tag_data, + { "Tag Data", "awdl.tag.data", + FT_BYTES, BASE_NONE, NULL, 0, + "Data Interpretation of tag", HFILL + } + }, + { &hf_awdl_tag_padding, + { "Padding (?)", "awdl.tag.padding", + FT_BYTES, BASE_NONE, NULL, 0, + "Unused (?) bytes at the end of the tag", HFILL + } + }, + + /* Version */ + { &hf_awdl_version, + { "AWDL Version", "awdl.version", + FT_UINT8, BASE_HEX, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_version_major, + { "AWDL Version Major", "awdl.version.major", + FT_UINT8, BASE_DEC, NULL, 0xf0, NULL, HFILL + } + }, + { &hf_awdl_version_minor, + { "AWDL Version Minor", "awdl.version.minor", + FT_UINT8, BASE_DEC, NULL, 0x0f, NULL, HFILL + } + }, + { &hf_awdl_version_devclass, + { "Device Class", "awdl.version.device_class", + FT_UINT8, BASE_DEC, VALS(awdl_version_devclass), 0, NULL, HFILL + } + }, + + /* Synchronization Tree */ + { &hf_awdl_synctree_addr, + { "Address", "awdl.synctree.addr", + FT_ETHER, BASE_NONE, NULL, 0x0, + "From tree root to leaf", HFILL + } + }, + + /* Data Path State */ + { &hf_awdl_datastate_flags, + { "Flags", "awdl.datastate.flags", + FT_UINT16, BASE_HEX, NULL, 0, + "Subsequent fields do not follow the order in which they appear in this bitmask", HFILL + } + }, + { &hf_awdl_datastate_flags_0, + { "Infrastructure BSSID and Channel", "awdl.datastate.flags.0", + FT_BOOLEAN, 16, NULL, 0x0001, NULL, HFILL + } + }, + { &hf_awdl_datastate_flags_1, + { "Infrastructure Address", "awdl.datastate.flags.1", + FT_BOOLEAN, 16, NULL, 0x0002, NULL, HFILL + } + }, + { &hf_awdl_datastate_flags_2, + { "AWDL Address", "awdl.datastate.flags.2", + FT_BOOLEAN, 16, NULL, 0x0004, NULL, HFILL + } + }, + { &hf_awdl_datastate_flags_3, + { "Bit 3", "awdl.datastate.flags.3", + FT_BOOLEAN, 16, NULL, 0x0008, NULL, HFILL + } + }, + { &hf_awdl_datastate_flags_4, + { "UMI", "awdl.datastate.flags.4", + FT_BOOLEAN, 16, NULL, 0x0010, NULL, HFILL + } + }, + { &hf_awdl_datastate_flags_5, + { "Bit 5", "awdl.datastate.flags.5", + FT_BOOLEAN, 16, NULL, 0x0020, NULL, HFILL + } + }, + { &hf_awdl_datastate_flags_6, + { "Is AirPlay (?)", "awdl.datastate.flags.6", + FT_BOOLEAN, 16, NULL, 0x0040, NULL, HFILL + } + }, + { &hf_awdl_datastate_flags_7, + { "Bit 7", "awdl.datastate.flags.7", + FT_BOOLEAN, 16, NULL, 0x0080, NULL, HFILL + } + }, + { &hf_awdl_datastate_flags_8, + { "Country Code", "awdl.datastate.flags.8", + FT_BOOLEAN, 16, NULL, 0x0100, NULL, HFILL + } + }, + { &hf_awdl_datastate_flags_9, + { "Social Channels", "awdl.datastate.flags.9", + FT_BOOLEAN, 16, NULL, 0x0200, NULL, HFILL + } + }, + { &hf_awdl_datastate_flags_10, + { "Bit 10", "awdl.datastate.flags.10", + FT_BOOLEAN, 16, NULL, 0x0400, NULL, HFILL + } + }, + { &hf_awdl_datastate_flags_11, + { "Bit 11", "awdl.datastate.flags.11", + FT_BOOLEAN, 16, NULL, 0x0800, NULL, HFILL + } + }, + { &hf_awdl_datastate_flags_12, + { "Unicast Options", "awdl.datastate.flags.12", + FT_BOOLEAN, 16, NULL, 0x1000, NULL, HFILL + } + }, + { &hf_awdl_datastate_flags_13, + { "Bit 13", "awdl.datastate.flags.13", + FT_BOOLEAN, 16, NULL, 0x2000, NULL, HFILL + } + }, + { &hf_awdl_datastate_flags_14, + { "Is Rangeable", "awdl.datastate.flags.14", + FT_BOOLEAN, 16, NULL, 0x4000, NULL, HFILL + } + }, + { &hf_awdl_datastate_flags_15, + { "Extension Flags", "awdl.datastate.flags.15", + FT_BOOLEAN, 16, NULL, 0x8000, NULL, HFILL + } + }, + { &hf_awdl_datastate_countrycode, + { "Country Code", "awdl.datastate.countrycode", + FT_STRING, BASE_NONE, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_datastate_social_channel, + { "Social Channel", "awdl.datastate.social_channel", + FT_UINT16, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_datastate_social_channel_map, + { "Social Channel Map", "awdl.datastate.social_channel_map", + FT_UINT16, BASE_HEX, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_datastate_social_channel_map_6, + { "Channel 6", "awdl.datastate.social_channel_map.ch6", + FT_BOOLEAN, 16, NULL, 0x1, NULL, HFILL + } + }, + { &hf_awdl_datastate_social_channel_map_44, + { "Channel 44", "awdl.datastate.social_channel_map.ch44", + FT_BOOLEAN, 16, NULL, 0x2, NULL, HFILL + } + }, + { &hf_awdl_datastate_social_channel_map_149, + { "Channel 149", "awdl.datastate.social_channel_map.ch149", + FT_BOOLEAN, 16, NULL, 0x4, NULL, HFILL + } + }, + { &hf_awdl_datastate_social_channel_map_unused, + { "Unused", "awdl.datastate.social_channel_map.unused", + FT_UINT16, BASE_HEX, NULL, 0xfff8, NULL, HFILL + } + }, + { &hf_awdl_datastate_infra_bssid, + { "Infrastructure BSSID", "awdl.datastate.infra_bssid", + FT_ETHER, BASE_NONE, NULL, 0x0, + "Address of the AP currently connected to", HFILL + } + }, + { &hf_awdl_datastate_infra_channel, + { "Infrastructure Channel", "awdl.datastate.infra_channel", + FT_UINT16, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_datastate_infra_address, + { "Infrastructure Address", "awdl.datastate.infra_addr", + FT_ETHER, BASE_NONE, NULL, 0x0, + "MAC address of this device", HFILL + } + }, + { &hf_awdl_datastate_awdl_address, + { "AWDL Address", "awdl.datastate.own_awdladdr", + FT_ETHER, BASE_NONE, NULL, 0x0, + "Randomized Address used for AWDL", HFILL + } + }, + { &hf_awdl_datastate_umi, + { "UMI (Airplay?)", "awdl.datastate.umi", + FT_UINT16, BASE_HEX_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_datastate_umioptions, + { "Unicast Options", "awdl.datastate.unicast_options", + FT_BYTES, BASE_NONE, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_datastate_umioptions_length, + { "Unicast Options Length", "awdl.datastate.unicast_options_length", + FT_UINT16, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_datastate_extflags, + { "Extended Flags", "awdl.datastate.extflags", + FT_UINT16, BASE_HEX, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_datastate_extflags_0, + { "Undecoded (?)", "awdl.datastate.extflags.0", + FT_BOOLEAN, 16, NULL, 0x1, NULL, HFILL + } + }, + { &hf_awdl_datastate_extflags_1, + { "Ranging Discovery", "awdl.datastate.extflags.1", + FT_BOOLEAN, 16, NULL, 0x2, NULL, HFILL + } + }, + { &hf_awdl_datastate_extflags_2, + { "Logtrigger ID", "awdl.datastate.extflags.2", + FT_BOOLEAN, 16, NULL, 0x4, NULL, HFILL + } + }, + { &hf_awdl_datastate_extflags_3to15, + { "Unknown", "awdl.datastate.extflags.3to15", + FT_UINT16, BASE_HEX_DEC, NULL, 0xfff8, NULL, HFILL + } + }, + { &hf_awdl_datastate_logtrigger, + { "Logtrigger ID", "awdl.datastate.logtrigger", + FT_UINT32, BASE_HEX, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_datastate_undecoded, + { "Undecoded", "awdl.datastate.undecoded", + FT_BYTES, BASE_NONE, NULL, 0, + "Possibly contains 'RLFC', a timestamp in ms, a AW sequence number, and 'payUpdateCounter'", HFILL + } + }, + + /* Arpa */ + { &hf_awdl_arpa, + { "Arpa", "awdl.arpa", + FT_NONE, BASE_NONE, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_arpa_name, + { "Host", "awdl.arpa.host", + FT_UINT_STRING, BASE_NONE, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_arpa_flags, + { "Flags", "awdl.arpa.flags", + FT_UINT8, BASE_HEX, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_arpa_short, + { "Domain (compressed)", "awdl.arpa.domain", + FT_UINT16, BASE_HEX, VALS(awdl_dns_compression), 0, NULL, HFILL + } + }, + + /* Synchronization Paramters */ + { &hf_awdl_syncparams_awcounter, + { "AW Sequence Number", "awdl.syncparams.awseqcounter", + FT_UINT16, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_syncparams_apbeaconalignment, + { "AP Beacon alignment delta", "awdl.syncparams.apbeaconalignment", + FT_UINT16, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_syncparams_master, + { "Master Address", "awdl.syncparams.master", + FT_ETHER, BASE_NONE, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_syncparams_tx_chan, + { "Next AW Channel", "awdl.syncparams.txchannel", + FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_syncparams_tx_counter, + { "Tx Counter", "awdl.syncparams.txcounter", + FT_UINT16, BASE_DEC | BASE_UNIT_STRING, &units_ieee80211_tu, 0x0, + "Time until next AW starts", HFILL + } + }, + { &hf_awdl_syncparams_master_chan, + { "Master Channel", "awdl.syncparams.masterchan", + FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_syncparams_guard_time, + { "Guard Time", "awdl.syncparams.guardtime", + FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_syncparams_aw_period, + { "Availability Window Period", "awdl.syncparams.awperiod", + FT_UINT16, BASE_DEC | BASE_UNIT_STRING, &units_ieee80211_tu, 0x0, NULL, HFILL + } + }, + { &hf_awdl_syncparams_action_frame_period, + { "Action Frame Period", "awdl.syncparams.afperiod", + FT_UINT16, BASE_DEC | BASE_UNIT_STRING, &units_ieee80211_tu, 0x0, NULL, HFILL + } + }, + { &hf_awdl_syncparams_awdl_flags, + { "AWDL Flags", "awdl.syncparams.awdlflags", + FT_UINT16, BASE_HEX, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_syncparams_aw_ext_length, + { "Availability Window Extension Length", "awdl.syncparams.aw.ext_len", + FT_UINT16, BASE_DEC | BASE_UNIT_STRING, &units_ieee80211_tu, 0x0, NULL, HFILL + } + }, + { &hf_awdl_syncparams_aw_cmn_length, + { "Availability Window Common Length", "awdl.syncparams.aw.common_len", + FT_UINT16, BASE_DEC | BASE_UNIT_STRING, &units_ieee80211_tu, 0x0, NULL, HFILL + } + }, + { &hf_awdl_syncparams_aw_remaining, + { "Remaining Availability Window Length", "awdl.syncparams.aw.remaining", + FT_INT16, BASE_DEC | BASE_UNIT_STRING, &units_ieee80211_tu, 0x0, NULL, HFILL + } + }, + { &hf_awdl_syncparams_ext_min, + { "Minimum Extension Count", "awdl.syncparams.ext.min", + FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_syncparams_ext_max_multi, + { "Maximum Extension Count for Multicast", "awdl.syncparams.ext.max_multicast", + FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_syncparams_ext_max_uni, + { "Maximum Extension Count for Unicast", "awdl.syncparams.ext.max_unicast", + FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_syncparams_ext_max_af, + { "Maximum Extension Count for Action Frame", "awdl.syncparams.ext.max_af", + FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_syncparams_presence_mode, + { "Presence Mode", "awdl.syncparams.presencemode", + FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL + } + }, + + /* Channel Sequence */ + { &hf_awdl_channelseq_channel_count, + { "Number of Channels (+1)", "awdl.channelseq.channels", + FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_channelseq_enc, + { "Encoding", "awdl.channelseq.encoding", + FT_UINT8, BASE_DEC, VALS(awdl_chanseq_enc), 0x0, NULL, HFILL + } + }, + { &hf_awdl_channelseq_duplicate, + { "Duplicate", "awdl.channelseq.duplicate", + FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_channelseq_step_count, + { "Step Count (+1)", "awdl.channelseq.step_count", + FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_channelseq_fill_chan, + { "Fill Channel", "awdl.channelseq.fill_channel", + FT_UINT16, BASE_HEX, VALS(awdl_chanseq_fill_chan), 0x0, NULL, HFILL + } + }, + { &hf_awdl_channelseq_channel_list, + { "Channel List", "awdl.channelseq.channel_list", + FT_NONE, BASE_NONE, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_channelseq_channel, + { "Channel", "awdl.channelseq.channel", + FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_channelseq_channel_number, + { "Channel Number", "awdl.channelseq.channel.number", + FT_UINT16, BASE_DEC, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_channelseq_channel_operating_class, + { "Operating Class", "awdl.channelseq.channel.operating_class", + FT_UINT8, BASE_HEX, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_channelseq_channel_flags, + { "Channel Flags", "awdl.channelseq.channel.flags", + FT_UINT8, BASE_HEX, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_channelseq_legacy_unused, + { "Unused", "awdl.channelseq.channel.unused", + FT_UINT8, BASE_DEC, NULL, 0xc0, NULL, HFILL + } + }, + { &hf_awdl_channelseq_legacy_band, + { "Band", "awdl.channelseq.channel.band", + FT_UINT8, BASE_DEC, VALS(awdl_chanseq_band), 0x30, NULL, HFILL + } + }, + { &hf_awdl_channelseq_legacy_bandwidth, + { "Bandwidth", "awdl.channelseq.channel.bandwidth", + FT_UINT8, BASE_DEC, VALS(awdl_chanseq_bandwidth), 0x0c, NULL, HFILL + } + }, + { &hf_awdl_channelseq_legacy_control_channel, + { "Control Channel", "awdl.channelseq.channel.control_channel", + FT_UINT8, BASE_DEC, VALS(awdl_chanseq_control_channel), 0x03, NULL, HFILL + } + }, + + /* Election Parameters */ + { &hf_awdl_electionparams_private_master, + { "Private Master Address", "awdl.electionparams.private.master", + FT_ETHER, BASE_NONE, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_electionparams_master, + { "Master Address", "awdl.electionparams.master", + FT_ETHER, BASE_NONE, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_electionparams_flags, + { "Flags", "awdl.electionparams.flags", + FT_UINT8, BASE_HEX, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_electionparams_id, + { "ID", "awdl.electionparams.id", + FT_UINT16, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_electionparams_distance, + { "Distance to Master", "awdl.electionparams.distance", + FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_electionparams_mastermetric, + { "Master Metric", "awdl.electionparams.mastermetric", + FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_electionparams_selfmetric, + { "Self Metric", "awdl.electionparams.selfmetric", + FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_electionparams_unknown, + { "Unknown", "awdl.electionparams.unknown", + FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_electionparams_private_mastermetric, + { "Private Master Metric", "awdl.electionparams.private.mastermetric", + FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_electionparams_private_id, + { "Private ID", "awdl.electionparams.private.id", + FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_electionparams_private_phc, + { "PHC", "awdl.electionparams.private.phc", + FT_UINT16, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + + /* Election Parameter v2 */ + { &hf_awdl_electionparams2_master, + { "Master Address", "awdl.electionparams2.master", + FT_ETHER, BASE_NONE, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_electionparams2_other, + { "Other Address", "awdl.electionparams2.other", + FT_ETHER, BASE_NONE, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_electionparams2_mastermetric, + { "Master Metric", "awdl.electionparams2.mastermetric", + FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_electionparams2_selfmetric, + { "Self Metric", "awdl.electionparams2.selfmetric", + FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_electionparams2_mastercounter, + { "Master Counter", "awdl.electionparams2.mastercounter", + FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_electionparams2_selfcounter, + { "Self Counter", "awdl.electionparams2.selfcounter", + FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_electionparams2_distance, + { "Distance to Master", "awdl.electionparams2.disstance", + FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_electionparams2_reserved, + { "Reserved", "awdl.electionparams2.reserved", + FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_electionparams2_unknown, + { "Unknown", "awdl.electionparams2.unknown", + FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + + /* Service Reponse */ + { &hf_awdl_dns_name_len, + { "Name Length", "awdl.dns.name.len", + FT_UINT16, BASE_DEC, NULL, 0x0, + "Includes length of type field", HFILL + } + }, + { &hf_awdl_dns_name, + { "Name", "awdl.dns.name", + FT_NONE, BASE_NONE, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_dns_name_label, + { "Label", "awdl.dns.name.label", + FT_UINT_STRING, BASE_NONE, NULL, 0x0, + "Part of a name", HFILL + } + }, + { &hf_awdl_dns_name_short, + { "Label (compressed)", "awdl.dns.name.compressed", + FT_UINT16, BASE_HEX, VALS(awdl_dns_compression), 0x0, + "Part of a name", HFILL + } + }, + { &hf_awdl_dns_type, + { "Type", "awdl.dns.type", + FT_UINT8, BASE_DEC, VALS(dns_types_vals), 0x0, NULL, HFILL + } + }, + { &hf_awdl_dns_data_len, + { "Data Length", "awdl.dns.data_len", + FT_UINT16, BASE_DEC, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_dns_txt, + { "TXT", "awdl.dns.txt", + FT_UINT_STRING, BASE_NONE, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_dns_ptr, + { "Domain Name", "awdl.dns.ptr", + FT_NONE, BASE_NONE, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_dns_ptr_label, + { "Label", "awdl.dns.ptr.label", + FT_UINT_STRING, BASE_NONE, NULL, 0x0, + "Part of a domain name", HFILL + } + }, + { &hf_awdl_dns_ptr_short, + { "Label (compressed)", "awdl.dns.ptr.short", + FT_UINT16, BASE_HEX, VALS(awdl_dns_compression), 0x0, + "Part of a domain name", HFILL + } + }, + { &hf_awdl_dns_target, + { "Target", "awdl.dns.target", + FT_NONE, BASE_NONE, NULL, 0x0, NULL, HFILL + } + }, + { &hf_awdl_dns_target_label, + { "Label", "awdl.dns.target.label", + FT_UINT_STRING, BASE_NONE, NULL, 0x0, + "Part of a target", HFILL + } + }, + { &hf_awdl_dns_target_short, + { "Label (compressed)", "awdl.dns.target.compressed", + FT_UINT16, BASE_HEX, VALS(awdl_dns_compression), 0x0, + "Part of a target", HFILL + } + }, + { &hf_awdl_dns_unknown, + { "Unknown", "awdl.dns.unknown", + FT_UINT16, BASE_DEC, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_dns_priority, + { "Priority", "awdl.dns.priority", + FT_UINT16, BASE_DEC, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_dns_weight, + { "Weight", "awdl.dns.weight", + FT_UINT16, BASE_DEC, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_dns_port, + { "Port", "awdl.dns.port", + FT_UINT16, BASE_DEC, NULL, 0, NULL, HFILL + } + }, + + /* Service Parameters */ + { &hf_awdl_serviceparams_sui, + { "SUI", "awdl.serviceparams.sui", + FT_UINT16, BASE_DEC, NULL, 0x0, + "Incremented by 1 for every service announcement change (should cause cache flush at receivers)", HFILL + } + }, + { &hf_awdl_serviceparams_enc_values, + { "Encoded Values", "awdl.serviceparams.values", + FT_NONE, BASE_NONE, NULL, 0, + "Encodes up to 256 unique 1-byte values. Calculation adds offsets to values.", HFILL + } + }, + { &hf_awdl_serviceparams_bitmask, + { "Offsets", "awdl.serviceparams.bitmask", + FT_UINT32, BASE_HEX, NULL, 0x0, + "Offset is 8*i if i-th bit is set", HFILL + } + }, + /* Generate with Python + * size = 32 + * for i in range(size): + * print('{{ &hf_awdl_serviceparams_bitmask_{},'.format(i)) + * print(' {{ "{}", "awdl.serviceparams.bitmask.{}",'.format(i, i)) + * print(' FT_BOOLEAN, {}, NULL, {}, NULL, HFILL'.format(size, hex(1 << i))) + * print(' }}'.format()) + * print('}},'.format()) + */ + { &hf_awdl_serviceparams_bitmask_0, + { "0", "awdl.serviceparams.bitmask.0", + FT_BOOLEAN, 32, NULL, 0x00000001, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_1, + { "1", "awdl.serviceparams.bitmask.1", + FT_BOOLEAN, 32, NULL, 0x00000002, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_2, + { "2", "awdl.serviceparams.bitmask.2", + FT_BOOLEAN, 32, NULL, 0x00000004, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_3, + { "3", "awdl.serviceparams.bitmask.3", + FT_BOOLEAN, 32, NULL, 0x00000008, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_4, + { "4", "awdl.serviceparams.bitmask.4", + FT_BOOLEAN, 32, NULL, 0x00000010, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_5, + { "5", "awdl.serviceparams.bitmask.5", + FT_BOOLEAN, 32, NULL, 0x00000020, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_6, + { "6", "awdl.serviceparams.bitmask.6", + FT_BOOLEAN, 32, NULL, 0x00000040, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_7, + { "7", "awdl.serviceparams.bitmask.7", + FT_BOOLEAN, 32, NULL, 0x00000080, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_8, + { "8", "awdl.serviceparams.bitmask.8", + FT_BOOLEAN, 32, NULL, 0x00000100, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_9, + { "9", "awdl.serviceparams.bitmask.9", + FT_BOOLEAN, 32, NULL, 0x00000200, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_10, + { "10", "awdl.serviceparams.bitmask.10", + FT_BOOLEAN, 32, NULL, 0x00000400, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_11, + { "11", "awdl.serviceparams.bitmask.11", + FT_BOOLEAN, 32, NULL, 0x00000800, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_12, + { "12", "awdl.serviceparams.bitmask.12", + FT_BOOLEAN, 32, NULL, 0x00001000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_13, + { "13", "awdl.serviceparams.bitmask.13", + FT_BOOLEAN, 32, NULL, 0x00002000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_14, + { "14", "awdl.serviceparams.bitmask.14", + FT_BOOLEAN, 32, NULL, 0x00004000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_15, + { "15", "awdl.serviceparams.bitmask.15", + FT_BOOLEAN, 32, NULL, 0x00008000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_16, + { "16", "awdl.serviceparams.bitmask.16", + FT_BOOLEAN, 32, NULL, 0x00010000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_17, + { "17", "awdl.serviceparams.bitmask.17", + FT_BOOLEAN, 32, NULL, 0x00020000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_18, + { "18", "awdl.serviceparams.bitmask.18", + FT_BOOLEAN, 32, NULL, 0x00040000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_19, + { "19", "awdl.serviceparams.bitmask.19", + FT_BOOLEAN, 32, NULL, 0x00080000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_20, + { "20", "awdl.serviceparams.bitmask.20", + FT_BOOLEAN, 32, NULL, 0x00100000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_21, + { "21", "awdl.serviceparams.bitmask.21", + FT_BOOLEAN, 32, NULL, 0x00200000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_22, + { "22", "awdl.serviceparams.bitmask.22", + FT_BOOLEAN, 32, NULL, 0x00400000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_23, + { "23", "awdl.serviceparams.bitmask.23", + FT_BOOLEAN, 32, NULL, 0x00800000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_24, + { "24", "awdl.serviceparams.bitmask.24", + FT_BOOLEAN, 32, NULL, 0x01000000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_25, + { "25", "awdl.serviceparams.bitmask.25", + FT_BOOLEAN, 32, NULL, 0x02000000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_26, + { "26", "awdl.serviceparams.bitmask.26", + FT_BOOLEAN, 32, NULL, 0x04000000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_27, + { "27", "awdl.serviceparams.bitmask.27", + FT_BOOLEAN, 32, NULL, 0x08000000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_28, + { "28", "awdl.serviceparams.bitmask.28", + FT_BOOLEAN, 32, NULL, 0x10000000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_29, + { "29", "awdl.serviceparams.bitmask.29", + FT_BOOLEAN, 32, NULL, 0x20000000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_30, + { "30", "awdl.serviceparams.bitmask.30", + FT_BOOLEAN, 32, NULL, 0x40000000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_bitmask_31, + { "31", "awdl.serviceparams.bitmask.31", + FT_BOOLEAN, 32, NULL, 0x80000000, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_values, + { "Values", "awdl.serviceparams.values", + FT_UINT8, BASE_HEX, NULL, 0, + "Value is i if i-th bit is set", HFILL + } + }, + { &hf_awdl_serviceparams_values_0, + { "0", "awdl.serviceparams.values.0", + FT_BOOLEAN, 8, NULL, 0x1, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_values_1, + { "1", "awdl.serviceparams.values.1", + FT_BOOLEAN, 8, NULL, 0x2, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_values_2, + { "2", "awdl.serviceparams.values.2", + FT_BOOLEAN, 8, NULL, 0x4, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_values_3, + { "3", "awdl.serviceparams.values.3", + FT_BOOLEAN, 8, NULL, 0x8, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_values_4, + { "4", "awdl.serviceparams.values.4", + FT_BOOLEAN, 8, NULL, 0x10, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_values_5, + { "5", "awdl.serviceparams.values.5", + FT_BOOLEAN, 8, NULL, 0x20, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_values_6, + { "6", "awdl.serviceparams.values.6", + FT_BOOLEAN, 8, NULL, 0x40, NULL, HFILL + } + }, + { &hf_awdl_serviceparams_values_7, + { "7", "awdl.serviceparams.values.7", + FT_BOOLEAN, 8, NULL, 0x80, NULL, HFILL + } + }, + + /* HT Capabilities */ + { &hf_awdl_ht_unknown, + { "Unknown", "awdl.ht.unknown", + FT_UINT16, BASE_HEX, NULL, 0, NULL, HFILL + } + }, + /* hf_ieee80211_* from packet-ieee80211.c */ + { &hf_awdl_ht_cap, + { "HT Capabilities Info", "awdl.ht.capabilities", + FT_UINT16, BASE_HEX, NULL, 0, "HT Capabilities information", HFILL + } + }, + { &hf_awdl_ht_ldpc_coding, + { "HT LDPC coding capability", "awdl.ht.capabilities.ldpccoding", + FT_BOOLEAN, 16, TFS(&ht_ldpc_coding_flag), 0x0001, NULL, HFILL + } + }, + { &hf_awdl_ht_chan_width, + { "HT Support channel width", "awdl.ht.capabilities.width", + FT_BOOLEAN, 16, TFS(&ht_chan_width_flag), 0x0002, NULL, HFILL + } + }, + { &hf_awdl_ht_sm_pwsave, + { "HT SM Power Save", "awdl.ht.capabilities.sm", + FT_UINT16, BASE_HEX, VALS(ht_sm_pwsave_flag), 0x000c, NULL, HFILL + } + }, + { &hf_awdl_ht_green, + { "HT Green Field", "awdl.ht.capabilities.green", + FT_BOOLEAN, 16, TFS(&ht_green_flag), 0x0010, NULL, HFILL + } + }, + { &hf_awdl_ht_short20, + { "HT Short GI for 20MHz", "awdl.ht.capabilities.short20", + FT_BOOLEAN, 16, TFS(&tfs_supported_not_supported), 0x0020, NULL, HFILL + } + }, + { &hf_awdl_ht_short40, + { "HT Short GI for 40MHz", "awdl.ht.capabilities.short40", + FT_BOOLEAN, 16, TFS(&tfs_supported_not_supported), 0x0040, NULL, HFILL + } + }, + { &hf_awdl_ht_tx_stbc, + { "HT Tx STBC", "awdl.ht.capabilities.txstbc", + FT_BOOLEAN, 16, TFS(&tfs_supported_not_supported), 0x0080, NULL, HFILL + } + }, + { &hf_awdl_ht_rx_stbc, + { "HT Rx STBC", "awdl.ht.capabilities.rxstbc", + FT_UINT16, BASE_HEX, VALS(ht_rx_stbc_flag), 0x0300, "HT Tx STBC", HFILL + } + }, + { &hf_awdl_ht_delayed_block_ack, + { "HT Delayed Block ACK", "awdl.ht.capabilities.delayedblockack", + FT_BOOLEAN, 16, TFS(&ht_delayed_block_ack_flag), 0x0400, NULL, HFILL + } + }, + { &hf_awdl_ht_max_amsdu, + { "HT Max A-MSDU length", "awdl.ht.capabilities.amsdu", + FT_BOOLEAN, 16, TFS(&ht_max_amsdu_flag), 0x0800, NULL, HFILL + } + }, + { &hf_awdl_ht_dss_cck_40, + { "HT DSSS/CCK mode in 40MHz", "awdl.ht.capabilities.dsscck", + FT_BOOLEAN, 16, TFS(&ht_dss_cck_40_flag), 0x1000, "HT DSS/CCK mode in 40MHz", HFILL + } + }, + { &hf_awdl_ht_psmp, + { "HT PSMP Support", "awdl.ht.capabilities.psmp", + FT_BOOLEAN, 16, TFS(&ht_psmp_flag), 0x2000, NULL, HFILL + } + }, + { &hf_awdl_ht_40_mhz_intolerant, + { "HT Forty MHz Intolerant", "awdl.ht.capabilities.40mhzintolerant", + FT_BOOLEAN, 16, TFS(&ht_40_mhz_intolerant_flag), 0x4000, NULL, HFILL + } + }, + { &hf_awdl_ht_l_sig, + { "HT L-SIG TXOP Protection support", "awdl.ht.capabilities.lsig", + FT_BOOLEAN, 16, TFS(&tfs_supported_not_supported), 0x8000, NULL, HFILL + } + }, + { &hf_awdl_ampduparam, + { "A-MPDU Parameters", "awdl.ht.ampduparam", + FT_UINT8, BASE_HEX, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_ampduparam_mpdu, + { "Maximum Rx A-MPDU Length", "awdl.ht.ampduparam.maxlength", + FT_UINT8, BASE_HEX, 0, 0x03, NULL, HFILL + } + }, + { &hf_awdl_ampduparam_mpdu_start_spacing, + { "MPDU Density", "awdl.ht.ampduparam.mpdudensity", + FT_UINT8, BASE_HEX, VALS(ampduparam_mpdu_start_spacing_flags), 0x1c, NULL, HFILL + } + }, + { &hf_awdl_ampduparam_reserved, + { "Reserved", "awdl.ht.ampduparam.reserved", + FT_UINT8, BASE_HEX, NULL, 0xE0, NULL, HFILL + } + }, + { &hf_awdl_mcsset, + { "Rx Supported Modulation and Coding Scheme Set", "awdl.ht.mcsset", + FT_NONE, BASE_NONE, NULL, 0, NULL, HFILL + } + }, + { &hf_awdl_mcsset_rx_bitmask, + { "Rx Modulation and Coding Scheme (One bit per modulation)", "awdl.ht.mcsset.rxbitmask", + FT_NONE, BASE_NONE, NULL, 0, "One bit per modulation", HFILL + } + }, + { &hf_awdl_mcsset_rx_bitmask_0to7, + { "Rx Bitmask Bits 0-7", "awdl.ht.mcsset.rxbitmask.0to7", + FT_UINT32, BASE_HEX, 0, 0x000000ff, NULL, HFILL + } + }, + { &hf_awdl_mcsset_rx_bitmask_8to15, + { "Rx Bitmask Bits 8-15", "awdl.ht.mcsset.rxbitmask.8to15", + FT_UINT32, BASE_HEX, 0, 0x0000ff00, NULL, HFILL + } + }, + { &hf_awdl_mcsset_rx_bitmask_16to23, + { "Rx Bitmask Bits 16-23", "awdl.ht.mcsset.rxbitmask.16to23", + FT_UINT32, BASE_HEX, 0, 0x00ff0000, NULL, HFILL + } + }, + { &hf_awdl_mcsset_rx_bitmask_24to31, + { "Rx Bitmask Bits 24-31", "awdl.ht.mcsset.rxbitmask.24to31", + FT_UINT32, BASE_HEX, 0, 0xff000000, NULL, HFILL + } + }, + }; + + static hf_register_info hf_apple_awdl_pid[] = { + { &hf_llc_apple_awdl_pid, + { "PID", "llc.apple_awdl_pid", + FT_UINT16, BASE_HEX, VALS(apple_awdl_pid_vals), 0x0, "Protocol ID", HFILL } + } + }; + + /* Setup protocol subtree array */ + static gint *ett[] = { + &ett_awdl_data, + &ett_awdl, + &ett_awdl_fixed_parameters, + &ett_awdl_tagged_parameters, + &ett_awdl_unknown, + &ett_awdl_tag, + &ett_awdl_channelseq_flags, + &ett_awdl_version, + &ett_awdl_dns_record, + &ett_awdl_dns_name, + &ett_awdl_channelseq_channel_list, + &ett_awdl_channelseq_channel, + &ett_awdl_datastate_flags, + &ett_awdl_datastate_social_channel_map, + &ett_awdl_datastate_extflags, + &ett_awdl_ht_capabilities, + &ett_awdl_ht_ampduparam, + &ett_awdl_ht_mcsset_tree, + &ett_awdl_ht_mcsbit_tree, + &ett_awdl_serviceparams_bitmask, + &ett_awdl_serviceparams_values, + &ett_awdl_serviceparams_value, + }; + + static ei_register_info ei[] = { + { &ei_awdl_tag_length, + { "awdl.tag.length.bad", PI_MALFORMED, PI_ERROR, + "Bad tag length", EXPFILL + } + }, + { &ei_awdl_tag_data, + { "awdl.tag.data.undecoded", PI_UNDECODED, PI_NOTE, + "Dissector for AWDL tag code not implemented", EXPFILL + } + }, + { &ei_awdl_dns_data_len, + { "awdl.dns.data_len.bad", PI_MALFORMED, PI_ERROR, + "Bad DNS data length", EXPFILL + } + }, + }; + + expert_module_t *expert_awdl; + + proto_awdl_data = proto_register_protocol("Apple Wireless Direct Link data frame", "AWDL data", "awdl_data"); + awdl_data_handle = register_dissector("awdl_data", dissect_awdl_data, proto_awdl_data); + + proto_awdl = proto_register_protocol("Apple Wireless Direct Link action frame", "AWDL", "awdl"); + awdl_action_handle = register_dissector("awdl", dissect_awdl_action, proto_awdl); + + expert_awdl = expert_register_protocol(proto_awdl); + expert_register_field_array(expert_awdl, ei, array_length(ei)); + + tagged_field_table = register_dissector_table("awdl.tag.number", "AWDL Tags", proto_awdl, FT_UINT8, BASE_DEC); + awdl_register_tags(); + + proto_register_field_array(proto_awdl_data, hf, array_length(hf)); + proto_register_subtree_array(ett, array_length(ett)); + + llc_add_oui(OUI_APPLE_AWDL, "llc.apple_awdl_pid", "LLC Apple AWDL OUI PID", hf_apple_awdl_pid, -1); +} + +void proto_reg_handoff_awdl(void) { + dissector_add_uint("wlan.action.vendor_specific", OUI_APPLE_AWDL, awdl_action_handle); + dissector_add_uint("llc.apple_awdl_pid", 0x0800, awdl_data_handle); + + ethertype_subdissector_table = find_dissector_table("ethertype"); +} + +/* + * Editor modelines - https://www.wireshark.org/tools/modelines.html + * + * Local variables: + * c-basic-offset: 2 + * tab-width: 8 + * indent-tabs-mode: nil + * End: + * + * vi: set shiftwidth=2 tabstop=8 expandtab: + * :indentSize=2:tabSize=8:noTabs=true: + */ |