summaryrefslogtreecommitdiffstats
path: root/epan/dissectors/packet-fc.c
diff options
context:
space:
mode:
Diffstat (limited to 'epan/dissectors/packet-fc.c')
-rw-r--r--epan/dissectors/packet-fc.c1600
1 files changed, 1600 insertions, 0 deletions
diff --git a/epan/dissectors/packet-fc.c b/epan/dissectors/packet-fc.c
new file mode 100644
index 00000000..39759145
--- /dev/null
+++ b/epan/dissectors/packet-fc.c
@@ -0,0 +1,1600 @@
+/* packet-fc.c
+ * Routines for Fibre Channel Decoding (FC Header, Link Ctl & Basic Link Svc)
+ * Copyright 2001, Dinesh G Dutt <ddutt@cisco.com>
+ * Copyright 2003 Ronnie Sahlberg, exchange first/last matching and
+ * tap listener and misc updates
+ *
+ * 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/exceptions.h>
+#include <epan/prefs.h>
+#include <epan/to_str.h>
+#include <wiretap/wtap.h>
+#include <epan/reassemble.h>
+#include <epan/conversation_table.h>
+#include <epan/srt_table.h>
+#include "packet-fc.h"
+#include "packet-fclctl.h"
+#include "packet-fcbls.h"
+#include <epan/crc32-tvb.h>
+#include <epan/expert.h>
+
+void proto_register_fc(void);
+void proto_reg_handoff_fc(void);
+
+#define FC_HEADER_SIZE 24
+#define FC_RCTL_VFT 0x50
+#define MDSHDR_TRAILER_SIZE 6
+
+/* Size of various fields in FC header in bytes */
+#define FC_RCTL_SIZE 1
+#define FC_DID_SIZE 3
+#define FC_CSCTL_SIZE 1
+#define FC_SID_SIZE 3
+#define FC_TYPE_SIZE 1
+#define FC_FCTL_SIZE 3
+#define FC_SEQID_SIZE 1
+#define FC_DFCTL_SIZE 1
+#define FC_SEQCNT_SIZE 2
+#define FC_OXID_SIZE 2
+#define FC_RXID_SIZE 2
+#define FC_PARAM_SIZE 4
+
+/* Initialize the protocol and registered fields */
+static int proto_fc = -1;
+static int hf_fc_time = -1;
+static int hf_fc_exchange_first_frame = -1;
+static int hf_fc_exchange_last_frame = -1;
+static int hf_fc_rctl = -1;
+static int hf_fc_did = -1;
+static int hf_fc_csctl = -1;
+static int hf_fc_sid = -1;
+static int hf_fc_id = -1;
+static int hf_fc_type = -1;
+static int hf_fc_fctl = -1;
+static int hf_fc_fctl_exchange_responder = -1;
+static int hf_fc_fctl_seq_recipient = -1;
+static int hf_fc_fctl_exchange_first = -1;
+static int hf_fc_fctl_exchange_last = -1;
+static int hf_fc_fctl_seq_last = -1;
+static int hf_fc_fctl_priority = -1;
+static int hf_fc_fctl_transfer_seq_initiative = -1;
+static int hf_fc_fctl_rexmitted_seq = -1;
+static int hf_fc_fctl_rel_offset = -1;
+static int hf_fc_fctl_abts_ack = -1;
+/* static int hf_fc_fctl_abts_not_ack = -1; */
+static int hf_fc_fctl_last_data_frame = -1;
+static int hf_fc_fctl_ack_0_1 = -1;
+static int hf_fc_seqid = -1;
+static int hf_fc_dfctl = -1;
+static int hf_fc_seqcnt = -1;
+static int hf_fc_oxid = -1;
+static int hf_fc_rxid = -1;
+static int hf_fc_param = -1;
+static int hf_fc_ftype = -1; /* Derived field, non-existent in FC hdr */
+static int hf_fc_reassembled = -1;
+static int hf_fc_relative_offset = -1;
+
+/* VFT fields */
+static int hf_fc_vft = -1;
+static int hf_fc_vft_rctl = -1;
+static int hf_fc_vft_ver = -1;
+static int hf_fc_vft_type = -1;
+static int hf_fc_vft_pri = -1;
+static int hf_fc_vft_vf_id = -1;
+static int hf_fc_vft_hop_ct = -1;
+
+/* Network_Header fields */
+static int hf_fc_nh_da = -1;
+static int hf_fc_nh_sa = -1;
+
+/* For Basic Link Svc */
+static int hf_fc_bls_seqid_vld = -1;
+static int hf_fc_bls_lastvld_seqid = -1;
+static int hf_fc_bls_oxid = -1;
+static int hf_fc_bls_rxid = -1;
+static int hf_fc_bls_lowseqcnt = -1;
+static int hf_fc_bls_hiseqcnt = -1;
+static int hf_fc_bls_rjtcode = -1;
+static int hf_fc_bls_rjtdetail = -1;
+static int hf_fc_bls_vendor = -1;
+
+/* For FC SOF */
+static int proto_fcsof = -1;
+
+static int hf_fcsof = -1;
+static int hf_fceof = -1;
+static int hf_fccrc = -1;
+static int hf_fccrc_status = -1;
+
+static int ett_fcsof = -1;
+static int ett_fceof = -1;
+static int ett_fccrc = -1;
+
+
+/* Initialize the subtree pointers */
+static gint ett_fc = -1;
+static gint ett_fctl = -1;
+static gint ett_fcbls = -1;
+static gint ett_fc_vft = -1;
+
+static expert_field ei_fccrc = EI_INIT;
+static expert_field ei_short_hdr = EI_INIT;
+/* static expert_field ei_frag_size = EI_INIT; */
+
+static dissector_handle_t fc_handle, fcsof_handle;
+static dissector_table_t fcftype_dissector_table;
+
+static int fc_tap = -1;
+
+typedef struct _fc_conv_data_t {
+ wmem_tree_t *exchanges;
+ wmem_tree_t *luns;
+} fc_conv_data_t;
+
+/* Reassembly stuff */
+static gboolean fc_reassemble = TRUE;
+static guint32 fc_max_frame_size = 1024;
+static reassembly_table fc_reassembly_table;
+
+typedef struct _fcseq_conv_key {
+ guint32 conv_idx;
+} fcseq_conv_key_t;
+
+typedef struct _fcseq_conv_data {
+ guint32 seq_cnt;
+} fcseq_conv_data_t;
+
+static wmem_map_t *fcseq_req_hash = NULL;
+
+/*
+ * Hash Functions
+ */
+static gint
+fcseq_equal(gconstpointer v, gconstpointer w)
+{
+ const fcseq_conv_key_t *v1 = (const fcseq_conv_key_t *)v;
+ const fcseq_conv_key_t *v2 = (const fcseq_conv_key_t *)w;
+
+ return (v1->conv_idx == v2->conv_idx);
+}
+
+static guint
+fcseq_hash (gconstpointer v)
+{
+ const fcseq_conv_key_t *key = (const fcseq_conv_key_t *)v;
+ guint val;
+
+ val = key->conv_idx;
+
+ return val;
+}
+
+static const char* fc_conv_get_filter_type(conv_item_t* conv, conv_filter_type_e filter)
+{
+ if ((filter == CONV_FT_SRC_ADDRESS) && (conv->src_address.type == AT_FC))
+ return "fc.s_id";
+
+ if ((filter == CONV_FT_DST_ADDRESS) && (conv->dst_address.type == AT_FC))
+ return "fc.d_id";
+
+ if ((filter == CONV_FT_ANY_ADDRESS) && (conv->src_address.type == AT_FC))
+ return "fc.id";
+
+ return CONV_FILTER_INVALID;
+}
+
+static ct_dissector_info_t fc_ct_dissector_info = {&fc_conv_get_filter_type};
+
+static tap_packet_status
+fc_conversation_packet(void *pct, packet_info *pinfo, epan_dissect_t *edt _U_, const void *vip, tap_flags_t flags)
+{
+ conv_hash_t *hash = (conv_hash_t*) pct;
+ hash->flags = flags;
+ const fc_hdr *fchdr=(const fc_hdr *)vip;
+
+ add_conversation_table_data(hash, &fchdr->s_id, &fchdr->d_id, 0, 0, 1, pinfo->fd->pkt_len, &pinfo->rel_ts, &pinfo->abs_ts, &fc_ct_dissector_info, CONVERSATION_NONE);
+
+ return TAP_PACKET_REDRAW;
+}
+
+static const char* fc_endpoint_get_filter_type(endpoint_item_t* endpoint, conv_filter_type_e filter)
+{
+ if ((filter == CONV_FT_ANY_ADDRESS) && (endpoint->myaddress.type == AT_FC))
+ return "fc.id";
+
+ return CONV_FILTER_INVALID;
+}
+
+static et_dissector_info_t fc_endpoint_dissector_info = {&fc_endpoint_get_filter_type};
+
+static tap_packet_status
+fc_endpoint_packet(void *pit, packet_info *pinfo, epan_dissect_t *edt _U_, const void *vip, tap_flags_t flags)
+{
+ conv_hash_t *hash = (conv_hash_t*) pit;
+ hash->flags = flags;
+ const fc_hdr *fchdr=(const fc_hdr *)vip;
+
+ /* Take two "add" passes per packet, adding for each direction, ensures that all
+ packets are counted properly (even if address is sending to itself)
+ XXX - this could probably be done more efficiently inside endpoint_table */
+ add_endpoint_table_data(hash, &fchdr->s_id, 0, TRUE, 1, pinfo->fd->pkt_len, &fc_endpoint_dissector_info, ENDPOINT_NONE);
+ add_endpoint_table_data(hash, &fchdr->d_id, 0, FALSE, 1, pinfo->fd->pkt_len, &fc_endpoint_dissector_info, ENDPOINT_NONE);
+
+ return TAP_PACKET_REDRAW;
+}
+
+#define FC_NUM_PROCEDURES 256
+
+static void
+fcstat_init(struct register_srt* srt _U_, GArray* srt_array)
+{
+ srt_stat_table *fc_srt_table;
+ guint32 i;
+
+ fc_srt_table = init_srt_table("Fibre Channel Types", NULL, srt_array, FC_NUM_PROCEDURES, NULL, "fc.type", NULL);
+ for (i = 0; i < FC_NUM_PROCEDURES; i++)
+ {
+ gchar* tmp_str = val_to_str_wmem(NULL, i, fc_fc4_val, "Unknown(0x%02x)");
+ init_srt_table_row(fc_srt_table, i, tmp_str);
+ wmem_free(NULL, tmp_str);
+ }
+}
+
+static tap_packet_status
+fcstat_packet(void *pss, packet_info *pinfo, epan_dissect_t *edt _U_, const void *prv, tap_flags_t flags _U_)
+{
+ guint i = 0;
+ srt_stat_table *fc_srt_table;
+ srt_data_t *data = (srt_data_t *)pss;
+ const fc_hdr *fc=(const fc_hdr *)prv;
+
+ /* we are only interested in reply packets */
+ if(!(fc->fctl&FC_FCTL_EXCHANGE_RESPONDER)){
+ return TAP_PACKET_DONT_REDRAW;
+ }
+ /* if we havnt seen the request, just ignore it */
+ if ( (!fc->fc_ex) || (fc->fc_ex->first_exchange_frame==0) ){
+ return TAP_PACKET_DONT_REDRAW;
+ }
+
+ fc_srt_table = g_array_index(data->srt_array, srt_stat_table*, i);
+ add_srt_table_data(fc_srt_table, fc->type, &fc->fc_ex->fc_time, pinfo);
+
+ return TAP_PACKET_REDRAW;
+}
+
+
+const value_string fc_fc4_val[] = {
+ {FC_TYPE_BLS, "Basic Link Svc"},
+ {FC_TYPE_ELS, "Ext Link Svc"},
+ {FC_TYPE_LLCSNAP, "LLC_SNAP"},
+ {FC_TYPE_IP, "IP/FC"},
+ {FC_TYPE_SCSI, "FCP"},
+ {FC_TYPE_FCCT, "FC_CT"},
+ {FC_TYPE_SWILS, "SW_ILS"},
+ {FC_TYPE_AL, "AL"},
+ {FC_TYPE_SNMP, "SNMP"},
+ {FC_TYPE_SB_FROM_CU, "SB-3(CU->Channel)"},
+ {FC_TYPE_SB_TO_CU, "SB-3(Channel->CU)"},
+ {0, NULL}
+};
+
+static const value_string fc_ftype_vals [] = {
+ {FC_FTYPE_UNDEF , "Unknown frame"},
+ {FC_FTYPE_SWILS, "SW_ILS"},
+ {FC_FTYPE_IP , "IP/FC"},
+ {FC_FTYPE_SCSI , "FCP"},
+ {FC_FTYPE_BLS , "Basic Link Svc"},
+ {FC_FTYPE_ELS , "ELS"},
+ {FC_FTYPE_FCCT , "FC_CT"},
+ {FC_FTYPE_LINKDATA, "Link Data"},
+ {FC_FTYPE_VDO, "Video Data"},
+ {FC_FTYPE_LINKCTL, "Link Ctl"},
+ {FC_FTYPE_SBCCS, "SBCCS"},
+ {FC_FTYPE_OHMS, "OHMS(Cisco MDS)"},
+ {0, NULL}
+};
+
+static const value_string fc_wka_vals[] _U_ = {
+ {FC_WKA_MULTICAST, "Multicast Server"},
+ {FC_WKA_CLKSYNC, "Clock Sync Server"},
+ {FC_WKA_KEYDIST, "Key Distribution Server"},
+ {FC_WKA_ALIAS, "Alias Server"},
+ {FC_WKA_QOSF, "QoS Facilitator"},
+ {FC_WKA_MGMT, "Management Server"},
+ {FC_WKA_TIME, "Time Server"},
+ {FC_WKA_DNS, "Directory Server"},
+ {FC_WKA_FABRIC_CTRLR, "Fabric Ctlr"},
+ {FC_WKA_FPORT, "F_Port Server"},
+ {FC_WKA_BCAST, "Broadcast ID"},
+ {0, NULL}
+};
+
+static const value_string fc_routing_val[] = {
+ {FC_RCTL_DEV_DATA, "Device_Data"},
+ {FC_RCTL_ELS, "Extended Link Services"},
+ {FC_RCTL_LINK_DATA, "FC-4 Link_Data"},
+ {FC_RCTL_VIDEO, "Video_Data"},
+ {FC_RCTL_BLS, "Basic Link Services"},
+ {FC_RCTL_LINK_CTL, "Link_Control Frame"},
+ {0, NULL}
+};
+
+static const value_string fc_iu_val[] = {
+ {FC_IU_UNCATEGORIZED , "Uncategorized Data"},
+ {FC_IU_SOLICITED_DATA , "Solicited Data"},
+ {FC_IU_UNSOLICITED_CTL , "Unsolicited Control"},
+ {FC_IU_SOLICITED_CTL , "Solicited Control"},
+ {FC_IU_UNSOLICITED_DATA, "Solicited Data"},
+ {FC_IU_DATA_DESCRIPTOR , "Data Descriptor"},
+ {FC_IU_UNSOLICITED_CMD , "Unsolicited Command"},
+ {FC_IU_CMD_STATUS , "Command Status"},
+ {0, NULL}
+};
+
+
+/* For FC SOF */
+#define FC_SOFC1 0xBCB51717
+#define FC_SOFI1 0xBCB55757
+#define FC_SOFN1 0xBCB53737
+#define FC_SOFI2 0xBCB55555
+#define FC_SOFN2 0xBCB53535
+#define FC_SOFI3 0xBCB55656
+#define FC_SOFN3 0xBCB53636
+#define FC_SOFC4 0xBCB51919
+#define FC_SOFI4 0xBCB55959
+#define FC_SOFN4 0xBCB53939
+#define FC_SOFF 0xBCB55858
+
+#define EOFT_NEG 0xBC957575
+#define EOFT_POS 0xBCB57575
+#define EOFDT_NEG 0xBC959595
+#define EOFDT_POS 0xBCB59595
+#define EOFA_NEG 0xBC95F5F5
+#define EOFA_POS 0xBCB5F5F5
+#define EOFN_NEG 0xBC95D5D5
+#define EOFN_POS 0xBCB5D5D5
+#define EOFNI_NEG 0xBC8AD5D5
+#define EOFNI_POS 0xBCAAD5D5
+#define EOFDTI_NEG 0xBC8A9595
+#define EOFDTI_POS 0xBCAA9595
+#define EOFRT_NEG 0xBC959999
+#define EOFRT_POS 0xBCB59999
+#define EOFRTI_NEG 0xBC8A9999
+#define EOFRTI_POS 0xBCAA9999
+
+static const value_string fc_sof_vals[] = {
+ {FC_SOFC1, "SOFc1 - SOF Connect Class 1 (Obsolete)" },
+ {FC_SOFI1, "SOFi1 - SOF Initiate Class 1 (Obsolete)" },
+ {FC_SOFN1, "SOFn1 - SOF Normal Class 1 (Obsolete)" },
+ {FC_SOFI2, "SOFi2 - SOF Initiate Class 2" },
+ {FC_SOFN2, "SOFn2 - SOF Normal Class 2" },
+ {FC_SOFI3, "SOFi3 - SOF Initiate Class 3" },
+ {FC_SOFN3, "SOFn3 - SOF Normal Class 3" },
+ {FC_SOFC4, "SOFc4 - SOF Activate Class 4 (Obsolete)" },
+ {FC_SOFI4, "SOFi4 - SOF Initiate Class 4 (Obsolete)" },
+ {FC_SOFN4, "SOFn4 - SOF Normal Class 4 (Obsolete)" },
+ {FC_SOFF, "SOFf - SOF Fabric" },
+ {0, NULL}
+};
+
+static const value_string fc_eof_vals[] = {
+ {EOFT_NEG, "EOFt- - EOF Terminate" },
+ {EOFT_POS, "EOFt+ - EOF Terminate" },
+ {EOFDT_NEG, "EOFdt- - EOF Disconnect-Terminate-Class 1 (Obsolete)" },
+ {EOFDT_POS, "EOFdt+ - EOF Disconnect-Terminate-Class 1 (Obsolete)" },
+ {EOFA_NEG, "EOFa- - EOF Abort" },
+ {EOFA_POS, "EOFa+ - EOF Abort" },
+ {EOFN_NEG, "EOFn- - EOF Normal" },
+ {EOFN_POS, "EOFn+ - EOF Normal" },
+ {EOFNI_NEG, "EOFni- - EOF Normal Invalid" },
+ {EOFNI_POS, "EOFni+ - EOF Normal Invalid" },
+ {EOFDTI_NEG, "EOFdti- - EOF Disconnect-Terminate-Invalid Class 1 (Obsolete)" },
+ {EOFDTI_POS, "EOFdti+ - EOF Disconnect-Terminate-Invalid Class 1 (Obsolete)" },
+ {EOFRT_NEG, "EOFrt- - EOF Remove-Terminate Class 4 (Obsolete)" },
+ {EOFRT_POS, "EOFrt+ - EOF Remove-Terminate Class 4 (Obsolete)" },
+ {EOFRTI_NEG, "EOFrti- - EOF Remove-Terminate Invalid Class 4 (Obsolete)" },
+ {EOFRTI_POS, "EOFrti+ - EOF Remove-Terminate Invalid Class 4 (Obsolete)" },
+ {0, NULL}
+};
+
+/* BA_ACC & BA_RJT are decoded in this file itself instead of a traditional
+ * dedicated file and dissector format because the dissector would require some
+ * fields of the FC_HDR such as param in some cases, type in some others, the
+ * lower 4 bits of r_ctl in some other cases etc. So, we decode BLS & Link Ctl
+ * in this file itself.
+ */
+static void
+dissect_fc_ba_acc (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ proto_tree *acc_tree;
+ int offset = 0;
+
+ /* Make entries in Protocol column and Info column on summary display */
+ col_set_str(pinfo->cinfo, COL_PROTOCOL, "BLS");
+
+ col_set_str(pinfo->cinfo, COL_INFO, "BA_ACC");
+
+ if (tree) {
+ acc_tree = proto_tree_add_subtree(tree, tvb, 0, -1, ett_fcbls, NULL, "Basic Link Svc");
+
+ proto_tree_add_item (acc_tree, hf_fc_bls_seqid_vld, tvb, offset++, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (acc_tree, hf_fc_bls_lastvld_seqid, tvb, offset++, 1, ENC_BIG_ENDIAN);
+ offset += 2; /* Skip reserved field */
+ proto_tree_add_item (acc_tree, hf_fc_bls_oxid, tvb, offset, 2, ENC_BIG_ENDIAN);
+ offset += 2;
+ proto_tree_add_item (acc_tree, hf_fc_bls_rxid, tvb, offset, 2, ENC_BIG_ENDIAN);
+ offset += 2;
+ proto_tree_add_item (acc_tree, hf_fc_bls_lowseqcnt, tvb, offset, 2, ENC_BIG_ENDIAN);
+ offset += 2;
+ proto_tree_add_item (acc_tree, hf_fc_bls_hiseqcnt, tvb, offset, 2, ENC_BIG_ENDIAN);
+ }
+}
+
+static void
+dissect_fc_ba_rjt (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ proto_tree *rjt_tree;
+ int offset = 0;
+
+ /* Make entries in Protocol column and Info column on summary display */
+ col_set_str(pinfo->cinfo, COL_PROTOCOL, "BLS");
+
+ col_set_str(pinfo->cinfo, COL_INFO, "BA_RJT");
+
+ if (tree) {
+ rjt_tree = proto_tree_add_subtree(tree, tvb, 0, -1, ett_fcbls, NULL, "Basic Link Svc");
+
+ proto_tree_add_item (rjt_tree, hf_fc_bls_rjtcode, tvb, offset+1, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rjt_tree, hf_fc_bls_rjtdetail, tvb, offset+2, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rjt_tree, hf_fc_bls_vendor, tvb, offset+3, 1, ENC_BIG_ENDIAN);
+ }
+}
+
+static guint8
+fc_get_ftype (guint8 r_ctl, guint8 type)
+{
+ /* A simple attempt to determine the upper level protocol based on the
+ * r_ctl & type fields.
+ */
+ switch (r_ctl & 0xF0) {
+ case FC_RCTL_DEV_DATA:
+ switch (type) {
+ case FC_TYPE_SWILS:
+ if ((r_ctl == 0x2) || (r_ctl == 0x3))
+ return FC_FTYPE_SWILS;
+ else
+ return FC_FTYPE_UNDEF;
+ case FC_TYPE_IP:
+ return FC_FTYPE_IP;
+ case FC_TYPE_SCSI:
+ return FC_FTYPE_SCSI;
+ case FC_TYPE_FCCT:
+ return FC_FTYPE_FCCT;
+ case FC_TYPE_SB_FROM_CU:
+ case FC_TYPE_SB_TO_CU:
+ return FC_FTYPE_SBCCS;
+ case FC_TYPE_VENDOR:
+ return FC_FTYPE_OHMS;
+ default:
+ return FC_FTYPE_UNDEF;
+ }
+ case FC_RCTL_ELS:
+ if (((r_ctl & 0x0F) == 0x2) || ((r_ctl & 0x0F) == 0x3))
+ return FC_FTYPE_ELS;
+ else if (type == FC_TYPE_ELS)
+ return FC_FTYPE_OHMS;
+ else
+ return FC_FTYPE_UNDEF;
+ case FC_RCTL_LINK_DATA:
+ switch (type) {
+ case FC_TYPE_SCSI:
+ return FC_FTYPE_SCSI;
+ default:
+ return FC_FTYPE_LINKDATA;
+ }
+ case FC_RCTL_VIDEO:
+ return FC_FTYPE_VDO;
+ case FC_RCTL_BLS:
+ if (type == 0)
+ return FC_FTYPE_BLS;
+ else
+ return FC_FTYPE_UNDEF;
+ case FC_RCTL_LINK_CTL:
+ return FC_FTYPE_LINKCTL;
+ default:
+ return FC_FTYPE_UNDEF;
+ }
+}
+
+static const value_string abts_ack_vals[] = {
+ {0x000000, "ABTS - Cont"},
+ {0x000010, "ABTS - Abort"},
+ {0x000020, "ABTS - Stop"},
+ {0x000030, "ABTS - Imm Seq Retx"},
+ {0,NULL}
+};
+#if 0
+static const value_string abts_not_ack_vals[] = {
+ {0x000000, "ABTS - Abort/MS"},
+ {0x000010, "ABTS - Abort/SS"},
+ {0x000020, "ABTS - Process/IB"},
+ {0x000030, "ABTS - Discard/MS/Imm Retx"},
+ {0,NULL}
+};
+#endif
+static const value_string last_data_frame_vals[] = {
+ {0x000000, "Last Data Frame - No Info"},
+ {0x004000, "Last Data Frame - Seq Imm"},
+ {0x008000, "Last Data Frame - Seq Soon"},
+ {0x00c000, "Last Data Frame - Seq Delyd"},
+ {0,NULL}
+};
+static const value_string ack_0_1_vals[] = {
+ {0x003000, "ACK_0 Required"},
+ {0x002000, "ACK_0 Required"},
+ {0x001000, "ACK_1 Required"},
+ {0x000000, "no ack required"},
+ {0,NULL}
+};
+static const true_false_string tfs_fc_fctl_exchange_responder = {
+ "Exchange Responder",
+ "Exchange Originator"
+};
+static const true_false_string tfs_fc_fctl_seq_recipient = {
+ "Seq Recipient",
+ "Seq Initiator"
+};
+static const true_false_string tfs_fc_fctl_exchange_first = {
+ "Exchg First",
+ "NOT exchg first"
+};
+static const true_false_string tfs_fc_fctl_exchange_last = {
+ "Exchg Last",
+ "NOT exchg last"
+};
+static const true_false_string tfs_fc_fctl_seq_last = {
+ "Seq Last",
+ "NOT seq last"
+};
+static const true_false_string tfs_fc_fctl_priority = {
+ "Priority",
+ "CS_CTL"
+};
+static const true_false_string tfs_fc_fctl_transfer_seq_initiative = {
+ "Transfer Seq Initiative",
+ "NOT transfer seq initiative"
+};
+static const true_false_string tfs_fc_fctl_rexmitted_seq = {
+ "Retransmitted Sequence",
+ "NOT retransmitted sequence"
+};
+static const true_false_string tfs_fc_fctl_rel_offset = {
+ "Rel Offset SET",
+ "Rel Offset NOT set"
+};
+
+/*
+ * Dissect the VFT header.
+ */
+static void
+dissect_fc_vft(proto_tree *parent_tree,
+ tvbuff_t *tvb, int offset)
+{
+ proto_item *item;
+ proto_tree *tree;
+ guint8 rctl;
+ guint8 ver;
+ guint8 type;
+ guint8 pri;
+ guint16 vf_id;
+ guint8 hop_ct;
+
+ rctl = tvb_get_guint8(tvb, offset);
+ type = tvb_get_guint8(tvb, offset + 1);
+ ver = (type >> 6) & 3;
+ type = (type >> 2) & 0xf;
+ vf_id = tvb_get_ntohs(tvb, offset + 2);
+ pri = (vf_id >> 13) & 7;
+ vf_id = (vf_id >> 1) & 0xfff;
+ hop_ct = tvb_get_guint8(tvb, offset + 4);
+
+ item = proto_tree_add_uint_format_value(parent_tree, hf_fc_vft, tvb, offset,
+ 8, vf_id, "VF_ID %d Pri %d Hop Count %d",
+ vf_id, pri, hop_ct);
+ tree = proto_item_add_subtree(item, ett_fc_vft);
+ proto_tree_add_uint(tree, hf_fc_vft_rctl, tvb, offset, 1, rctl);
+ proto_tree_add_uint(tree, hf_fc_vft_ver, tvb, offset + 1, 1, ver);
+ proto_tree_add_uint(tree, hf_fc_vft_type, tvb, offset + 1, 1, type);
+ proto_tree_add_uint(tree, hf_fc_vft_pri, tvb, offset + 2, 1, pri);
+ proto_tree_add_uint(tree, hf_fc_vft_vf_id, tvb, offset + 2, 2, vf_id);
+ proto_tree_add_uint(tree, hf_fc_vft_hop_ct, tvb, offset + 4, 1, hop_ct);
+}
+
+/* code to dissect the F_CTL bitmask */
+static void
+dissect_fc_fctl(packet_info *pinfo _U_, proto_tree *parent_tree, tvbuff_t *tvb, int offset)
+{
+ static int * const flags[] = {
+ &hf_fc_fctl_exchange_responder,
+ &hf_fc_fctl_seq_recipient,
+ &hf_fc_fctl_exchange_first,
+ &hf_fc_fctl_exchange_last,
+ &hf_fc_fctl_seq_last,
+ &hf_fc_fctl_priority,
+ &hf_fc_fctl_transfer_seq_initiative,
+ &hf_fc_fctl_last_data_frame,
+ &hf_fc_fctl_ack_0_1,
+ &hf_fc_fctl_rexmitted_seq,
+ &hf_fc_fctl_abts_ack,
+ &hf_fc_fctl_rel_offset,
+ NULL
+ };
+
+ proto_tree_add_bitmask_with_flags(parent_tree, tvb, offset, hf_fc_fctl,
+ ett_fctl, flags, ENC_BIG_ENDIAN, BMT_NO_INT);
+}
+
+static const value_string fc_bls_proto_val[] = {
+ {FC_BLS_NOP , "NOP"},
+ {FC_BLS_ABTS , "ABTS"},
+ {FC_BLS_RMC , "RMC"},
+ {FC_BLS_BAACC , "BA_ACC"},
+ {FC_BLS_BARJT , "BA_RJT"},
+ {FC_BLS_PRMT , "PRMT"},
+ {0, NULL}
+};
+
+static const value_string fc_els_proto_val[] = {
+ {0x01 , "Solicited Data"},
+ {0x02 , "Request"},
+ {0x03 , "Reply"},
+ {0, NULL}
+};
+
+/* Code to actually dissect the packets */
+static void
+dissect_fc_helper (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, gboolean is_ifcp, fc_data_t* fc_data)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ proto_item *ti, *hidden_item;
+ proto_tree *fc_tree;
+ tvbuff_t *next_tvb;
+ int offset = 0, next_offset;
+ int vft_offset = -1;
+ gboolean is_lastframe_inseq, is_1frame_inseq, is_exchg_resp = 0;
+ fragment_head *fcfrag_head;
+ guint32 frag_id, frag_size;
+ guint8 df_ctl, seq_id;
+ guint32 f_ctl;
+ address addr;
+
+ guint32 param, exchange_key;
+ guint16 real_seqcnt;
+ guint8 ftype;
+
+ fc_hdr* fchdr = wmem_new(pinfo->pool, fc_hdr); /* Needed by conversations, not just tap */
+ fc_exchange_t *fc_ex;
+ fc_conv_data_t *fc_conv_data=NULL;
+
+ conversation_t *conversation;
+ fcseq_conv_data_t *cdata;
+ fcseq_conv_key_t ckey, *req_key;
+
+ /* Make entries in Protocol column and Info column on summary display */
+ col_set_str(pinfo->cinfo, COL_PROTOCOL, "FC");
+
+ fchdr->r_ctl = tvb_get_guint8 (tvb, offset);
+ fchdr->fc_ex = NULL;
+
+ /*
+ * If the frame contains a VFT (virtual fabric tag), decode it
+ * as a separate header before the FC frame header.
+ *
+ * This used to be called the Cisco-proprietary EISL field, but is now
+ * standardized in FC-FS-2. See section 10.2.4.
+ */
+ if (fchdr->r_ctl == FC_RCTL_VFT) {
+ vft_offset = offset;
+ offset += 8;
+ fchdr->r_ctl = tvb_get_guint8 (tvb, offset);
+ }
+
+ /* Each fc endpoint pair gets its own TCP session in iFCP but
+ * the src/dst ids are undefined(==semi-random) in the FC header.
+ * This means we can no track conversations for FC over iFCP by using
+ * the FC src/dst addresses.
+ * For iFCP: Do not update the pinfo src/dst struct and let it remain
+ * being tcpip src/dst so that request/response matching in the FCP layer
+ * will use ip addresses instead and still work.
+ */
+ if(!is_ifcp){
+ set_address_tvb (&pinfo->dst, AT_FC, 3, tvb, offset+1);
+ set_address_tvb (&pinfo->src, AT_FC, 3, tvb, offset+5);
+ conversation_set_conv_addr_port_endpoints(pinfo, &pinfo->src, &pinfo->dst, CONVERSATION_EXCHG, 0, 0);
+ } else {
+ conversation_set_conv_addr_port_endpoints(pinfo, &pinfo->src, &pinfo->dst, CONVERSATION_EXCHG, pinfo->srcport, pinfo->destport);
+ }
+ set_address(&fchdr->d_id, pinfo->dst.type, pinfo->dst.len, pinfo->dst.data);
+ set_address(&fchdr->s_id, pinfo->src.type, pinfo->src.len, pinfo->src.data);
+
+ fchdr->cs_ctl = tvb_get_guint8 (tvb, offset+4);
+ fchdr->type = tvb_get_guint8 (tvb, offset+8);
+ fchdr->fctl=tvb_get_ntoh24(tvb,offset+9);
+ fchdr->seqcnt = tvb_get_ntohs (tvb, offset+14);
+ fchdr->oxid=tvb_get_ntohs(tvb,offset+16);
+ fchdr->rxid=tvb_get_ntohs(tvb,offset+18);
+ fchdr->relative_offset=0;
+ param = tvb_get_ntohl (tvb, offset+20);
+ seq_id = tvb_get_guint8 (tvb, offset+12);
+
+ /* set up a conversation and conversation data */
+ /* TODO treat the fc address s_id==00.00.00 as a wildcard matching anything */
+ conversation=find_or_create_conversation(pinfo);
+ fc_conv_data=(fc_conv_data_t *)conversation_get_proto_data(conversation, proto_fc);
+ if(!fc_conv_data){
+ fc_conv_data=wmem_new(wmem_file_scope(), fc_conv_data_t);
+ fc_conv_data->exchanges=wmem_tree_new(wmem_file_scope());
+ fc_conv_data->luns=wmem_tree_new(wmem_file_scope());
+ conversation_add_proto_data(conversation, proto_fc, fc_conv_data);
+ }
+
+ /* Set up LUN data. OXID + LUN make up unique exchanges, but LUN is populated in subdissectors
+ and not necessarily in every frame. Stub it here for now */
+ fchdr->lun = 0xFFFF;
+ if (pinfo->fd->visited) {
+ fchdr->lun = (guint16)GPOINTER_TO_UINT(wmem_tree_lookup32(fc_conv_data->luns, fchdr->oxid));
+ }
+
+ /* In the interest of speed, if "tree" is NULL, don't do any work not
+ necessary to generate protocol tree items. */
+ ti = proto_tree_add_protocol_format (tree, proto_fc, tvb, offset, FC_HEADER_SIZE, "Fibre Channel");
+ fc_tree = proto_item_add_subtree (ti, ett_fc);
+
+ /*is_ack = ((fchdr->r_ctl == 0xC0) || (fchdr->r_ctl == 0xC1));*/
+
+ /* There are two ways to determine if this is the first frame of a
+ * sequence. Either:
+ * (i) The SOF bits indicate that this is the first frame OR
+ * (ii) This is an SOFf frame and seqcnt is 0.
+ */
+ is_1frame_inseq = (((fc_data->sof_eof & FC_DATA_SOF_FIRST_FRAME) == FC_DATA_SOF_FIRST_FRAME) ||
+ (((fc_data->sof_eof & FC_DATA_SOF_SOFF) == FC_DATA_SOF_SOFF) &&
+ (fchdr->seqcnt == 0)));
+
+ is_lastframe_inseq = ((fc_data->sof_eof & FC_DATA_EOF_LAST_FRAME) == FC_DATA_EOF_LAST_FRAME);
+
+ is_lastframe_inseq |= fchdr->fctl & FC_FCTL_SEQ_LAST;
+ /*is_valid_frame = ((pinfo->sof_eof & 0x40) == 0x40);*/
+
+ ftype = fc_get_ftype (fchdr->r_ctl, fchdr->type);
+
+ col_add_str (pinfo->cinfo, COL_INFO, val_to_str (ftype, fc_ftype_vals,
+ "Unknown Type (0x%x)"));
+
+ if (ftype == FC_FTYPE_LINKCTL)
+ col_append_fstr (pinfo->cinfo, COL_INFO, ", %s",
+ val_to_str ((fchdr->r_ctl & 0x0F),
+ fc_lctl_proto_val,
+ "LCTL 0x%x"));
+
+ if (vft_offset >= 0) {
+ dissect_fc_vft(fc_tree, tvb, vft_offset);
+ }
+ switch (fchdr->r_ctl & 0xF0) {
+
+ case FC_RCTL_DEV_DATA:
+ case FC_RCTL_LINK_DATA:
+ case FC_RCTL_VIDEO:
+ /* the lower 4 bits of R_CTL are the information category */
+ proto_tree_add_uint_format_value(fc_tree, hf_fc_rctl, tvb, offset,
+ FC_RCTL_SIZE, fchdr->r_ctl,
+ "0x%x(%s/%s)",
+ fchdr->r_ctl,
+ val_to_str ((fchdr->r_ctl & 0xF0),
+ fc_routing_val, "0x%x"),
+ val_to_str ((fchdr->r_ctl & 0x0F),
+ fc_iu_val, "0x%x"));
+ break;
+
+ case FC_RCTL_LINK_CTL:
+ /* the lower 4 bits of R_CTL indicate the type of link ctl frame */
+ proto_tree_add_uint_format_value(fc_tree, hf_fc_rctl, tvb, offset,
+ FC_RCTL_SIZE, fchdr->r_ctl,
+ "0x%x(%s/%s)",
+ fchdr->r_ctl,
+ val_to_str ((fchdr->r_ctl & 0xF0),
+ fc_routing_val, "0x%x"),
+ val_to_str ((fchdr->r_ctl & 0x0F),
+ fc_lctl_proto_val, "0x%x"));
+ break;
+
+ case FC_RCTL_BLS:
+ switch (fchdr->type) {
+
+ case 0x00:
+ /* the lower 4 bits of R_CTL indicate the type of BLS frame */
+ proto_tree_add_uint_format_value(fc_tree, hf_fc_rctl, tvb, offset,
+ FC_RCTL_SIZE, fchdr->r_ctl,
+ "0x%x(%s/%s)",
+ fchdr->r_ctl,
+ val_to_str ((fchdr->r_ctl & 0xF0),
+ fc_routing_val, "0x%x"),
+ val_to_str ((fchdr->r_ctl & 0x0F),
+ fc_bls_proto_val, "0x%x"));
+ break;
+
+ default:
+ proto_tree_add_uint_format_value(fc_tree, hf_fc_rctl, tvb, offset,
+ FC_RCTL_SIZE, fchdr->r_ctl,
+ "0x%x(%s/0x%x)",
+ fchdr->r_ctl,
+ val_to_str ((fchdr->r_ctl & 0xF0),
+ fc_routing_val, "0x%x"),
+ fchdr->r_ctl & 0x0F);
+ break;
+ }
+ break;
+
+ case FC_RCTL_ELS:
+ switch (fchdr->type) {
+
+ case 0x01:
+ /* the lower 4 bits of R_CTL indicate the type of ELS frame */
+ proto_tree_add_uint_format_value(fc_tree, hf_fc_rctl, tvb, offset,
+ FC_RCTL_SIZE, fchdr->r_ctl,
+ "0x%x(%s/%s)",
+ fchdr->r_ctl,
+ val_to_str ((fchdr->r_ctl & 0xF0),
+ fc_routing_val, "0x%x"),
+ val_to_str ((fchdr->r_ctl & 0x0F),
+ fc_els_proto_val, "0x%x"));
+ break;
+
+ default:
+ proto_tree_add_uint_format_value(fc_tree, hf_fc_rctl, tvb, offset,
+ FC_RCTL_SIZE, fchdr->r_ctl,
+ "0x%x(%s/0x%x)",
+ fchdr->r_ctl,
+ val_to_str ((fchdr->r_ctl & 0xF0),
+ fc_routing_val, "0x%x"),
+ fchdr->r_ctl & 0x0F);
+ break;
+ }
+ break;
+
+ default:
+ proto_tree_add_uint_format_value(fc_tree, hf_fc_rctl, tvb, offset,
+ FC_RCTL_SIZE, fchdr->r_ctl,
+ "0x%x(%s/0x%x)",
+ fchdr->r_ctl,
+ val_to_str ((fchdr->r_ctl & 0xF0),
+ fc_routing_val, "0x%x"),
+ fchdr->r_ctl & 0x0F);
+ break;
+ }
+
+ hidden_item = proto_tree_add_uint (fc_tree, hf_fc_ftype, tvb, offset, 1,
+ ftype);
+ proto_item_set_hidden(hidden_item);
+
+ /* XXX - use "fc_wka_vals[]" on this? */
+ set_address(&addr, AT_FC, 3, fchdr->d_id.data);
+ proto_tree_add_item(fc_tree, hf_fc_did, tvb, offset+1, 3, ENC_NA);
+ hidden_item = proto_tree_add_item (fc_tree, hf_fc_id, tvb, offset+1, 3, ENC_NA);
+ proto_item_set_hidden(hidden_item);
+
+ proto_tree_add_uint (fc_tree, hf_fc_csctl, tvb, offset+4, 1, fchdr->cs_ctl);
+
+ /* XXX - use "fc_wka_vals[]" on this? */
+ set_address(&addr, AT_FC, 3, fchdr->s_id.data);
+ proto_tree_add_item(fc_tree, hf_fc_sid, tvb, offset+5, 3, ENC_NA);
+ hidden_item = proto_tree_add_item (fc_tree, hf_fc_id, tvb, offset+5, 3, ENC_NA);
+ proto_item_set_hidden(hidden_item);
+
+ if (ftype == FC_FTYPE_LINKCTL) {
+ if (((fchdr->r_ctl & 0x0F) == FC_LCTL_FBSYB) ||
+ ((fchdr->r_ctl & 0x0F) == FC_LCTL_FBSYL)) {
+ /* for F_BSY frames, the upper 4 bits of the type field specify the
+ * reason for the BSY.
+ */
+ proto_tree_add_uint_format_value(fc_tree, hf_fc_type, tvb,
+ offset+8, FC_TYPE_SIZE,
+ fchdr->type,"0x%x(%s)", fchdr->type,
+ fclctl_get_typestr ((guint8) (fchdr->r_ctl & 0x0F),
+ fchdr->type));
+ } else {
+ proto_tree_add_item (fc_tree, hf_fc_type, tvb, offset+8, 1, ENC_BIG_ENDIAN);
+ }
+ } else {
+ proto_tree_add_item (fc_tree, hf_fc_type, tvb, offset+8, 1, ENC_BIG_ENDIAN);
+ }
+
+
+ dissect_fc_fctl(pinfo, fc_tree, tvb, offset+9);
+ f_ctl = tvb_get_ntoh24(tvb, offset+9);
+
+
+ proto_tree_add_item (fc_tree, hf_fc_seqid, tvb, offset+12, 1, ENC_BIG_ENDIAN);
+
+ df_ctl = tvb_get_guint8(tvb, offset+13);
+
+ proto_tree_add_uint (fc_tree, hf_fc_dfctl, tvb, offset+13, 1, df_ctl);
+ proto_tree_add_uint (fc_tree, hf_fc_seqcnt, tvb, offset+14, 2, fchdr->seqcnt);
+ proto_tree_add_uint (fc_tree, hf_fc_oxid, tvb, offset+16, 2, fchdr->oxid);
+ proto_tree_add_uint (fc_tree, hf_fc_rxid, tvb, offset+18, 2, fchdr->rxid);
+
+ if (ftype == FC_FTYPE_LINKCTL) {
+ if (((fchdr->r_ctl & 0x0F) == FC_LCTL_FRJT) ||
+ ((fchdr->r_ctl & 0x0F) == FC_LCTL_PRJT) ||
+ ((fchdr->r_ctl & 0x0F) == FC_LCTL_PBSY)) {
+ /* In all these cases of Link Ctl frame, the parameter field
+ * encodes the detailed error message
+ */
+ proto_tree_add_uint_format_value(fc_tree, hf_fc_param, tvb,
+ offset+20, 4, param,
+ "0x%x(%s)", param,
+ fclctl_get_paramstr (pinfo->pool, (fchdr->r_ctl & 0x0F),
+ param));
+ } else {
+ proto_tree_add_item (fc_tree, hf_fc_param, tvb, offset+20, 4, ENC_BIG_ENDIAN);
+ }
+ } else if (ftype == FC_FTYPE_BLS) {
+ if ((fchdr->r_ctl & 0x0F) == FC_BLS_ABTS) {
+ proto_tree_add_uint_format_value(fc_tree, hf_fc_param, tvb,
+ offset+20, 4, param,
+ "0x%x(%s)", param,
+ ((param & 0x0F) == 1 ? "Abort Sequence" :
+ "Abort Exchange"));
+ } else {
+ proto_tree_add_item (fc_tree, hf_fc_param, tvb, offset+20,
+ 4, ENC_BIG_ENDIAN);
+ }
+ } else if (ftype == FC_FTYPE_SCSI ) {
+ if (f_ctl&FC_FCTL_REL_OFFSET){
+ proto_tree_add_item (fc_tree, hf_fc_relative_offset, tvb, offset+20, 4, ENC_BIG_ENDIAN);
+ fchdr->relative_offset=tvb_get_ntohl(tvb, offset+20);
+ } else {
+ proto_tree_add_item (fc_tree, hf_fc_param, tvb, offset+20, 4, ENC_BIG_ENDIAN);
+ }
+ } else {
+ proto_tree_add_item (fc_tree, hf_fc_param, tvb, offset+20, 4, ENC_BIG_ENDIAN);
+ }
+
+ /* Skip the Frame_Header */
+ next_offset = offset + FC_HEADER_SIZE;
+
+ /* Network_Header present? */
+ if (df_ctl & FC_DFCTL_NH) {
+ proto_tree_add_item(fc_tree, hf_fc_nh_da, tvb, next_offset, 8, ENC_NA);
+ proto_tree_add_item(fc_tree, hf_fc_nh_sa, tvb, next_offset+8, 8, ENC_NA);
+ next_offset += 16;
+ }
+
+ /* XXX - handle Association_Header and Device_Header here */
+
+ if (ftype == FC_FTYPE_LINKCTL) {
+ /* ACK_1 frames and other LINK_CTL frames echo the last seq bit if the
+ * packet they're ack'ing did not have it set. So, we'll incorrectly
+ * flag them as being fragmented when they're not. This fixes the
+ * problem
+ */
+ is_lastframe_inseq = TRUE;
+ } else {
+ is_exchg_resp = (f_ctl & FC_FCTL_EXCHANGE_RESPONDER) != 0;
+ }
+
+ if (tvb_reported_length (tvb) < FC_HEADER_SIZE) {
+ proto_tree_add_expert(fc_tree, pinfo, &ei_short_hdr,
+ tvb, 0, tvb_reported_length(tvb));
+ return;
+ }
+
+ frag_size = tvb_reported_length (tvb)-FC_HEADER_SIZE;
+
+ /* If there is an MDS header, we need to subtract the MDS trailer size
+ * Link Ctl, BLS & OHMS are all (encap header + FC Header + encap trailer)
+ * and are never fragmented and so we ignore the frag_size assertion for
+ * these frames.
+ */
+ if (fc_data->ethertype == ETHERTYPE_FCFT) {
+ if ((frag_size < MDSHDR_TRAILER_SIZE) ||
+ ((frag_size == MDSHDR_TRAILER_SIZE) && (ftype != FC_FTYPE_LINKCTL) &&
+ (ftype != FC_FTYPE_BLS) && (ftype != FC_FTYPE_OHMS))) {
+ proto_tree_add_expert(fc_tree, pinfo, &ei_short_hdr,
+ tvb, FC_HEADER_SIZE, frag_size);
+ return;
+ }
+ frag_size -= MDSHDR_TRAILER_SIZE;
+ } else if (fc_data->ethertype == ETHERTYPE_BRDWALK) {
+ if (frag_size <= 8) {
+ proto_tree_add_expert(fc_tree, pinfo, &ei_short_hdr,
+ tvb, FC_HEADER_SIZE, frag_size);
+ return;
+ }
+ frag_size -= 8; /* 4 byte of FC CRC +
+ 4 bytes of error+EOF = 8 bytes */
+ }
+
+ if (!is_lastframe_inseq) {
+ /* Show this only as a fragmented FC frame */
+ col_append_str (pinfo->cinfo, COL_INFO, " (Fragmented)");
+ }
+
+ /* If this is a fragment, attempt to check if fully reassembled frame is
+ * present, if we're configured to reassemble.
+ */
+ if ((ftype != FC_FTYPE_LINKCTL) && (ftype != FC_FTYPE_BLS) &&
+ (ftype != FC_FTYPE_OHMS) &&
+ (!is_lastframe_inseq || !is_1frame_inseq) && fc_reassemble &&
+ tvb_bytes_exist(tvb, FC_HEADER_SIZE, frag_size) && tree) {
+ /* Add this to the list of fragments */
+
+ /* In certain cases such as FICON, the SEQ_CNT is streaming
+ * i.e. continuously increasing. So, zero does not signify the
+ * first frame of the sequence. To fix this, we need to save the
+ * SEQ_CNT of the first frame in sequence and use this value to
+ * determine the actual offset into a frame.
+ */
+ ckey.conv_idx = conversation->conv_index;
+
+ cdata = (fcseq_conv_data_t *)wmem_map_lookup (fcseq_req_hash,
+ &ckey);
+
+ if (is_1frame_inseq) {
+ if (cdata) {
+ /* Since we never free the memory used by an exchange, this maybe a
+ * case of another request using the same exchange as a previous
+ * req.
+ */
+ cdata->seq_cnt = fchdr->seqcnt;
+ }
+ else {
+ req_key = wmem_new(wmem_file_scope(), fcseq_conv_key_t);
+ req_key->conv_idx = conversation->conv_index;
+
+ cdata = wmem_new(wmem_file_scope(), fcseq_conv_data_t);
+ cdata->seq_cnt = fchdr->seqcnt;
+
+ wmem_map_insert (fcseq_req_hash, req_key, cdata);
+ }
+ real_seqcnt = 0;
+ }
+ else if (cdata != NULL) {
+ real_seqcnt = fchdr->seqcnt - cdata->seq_cnt ;
+ }
+ else {
+ real_seqcnt = fchdr->seqcnt;
+ }
+
+ /* Verify that this is a valid fragment */
+ if (is_lastframe_inseq && !is_1frame_inseq && !real_seqcnt) {
+ /* This is a frame that purports to be the last frame in a
+ * sequence, is not the first frame, but has a seqcnt that is
+ * 0. This is a bogus frame, don't attempt to reassemble it.
+ */
+ next_tvb = tvb_new_subset_remaining (tvb, next_offset);
+ col_append_str (pinfo->cinfo, COL_INFO, " (Bogus Fragment)");
+ } else {
+
+ frag_id = ((fchdr->oxid << 16) ^ seq_id) | is_exchg_resp ;
+
+ /* We assume that all frames are of the same max size */
+ fcfrag_head = fragment_add (&fc_reassembly_table,
+ tvb, FC_HEADER_SIZE,
+ pinfo, frag_id, NULL,
+ real_seqcnt * fc_max_frame_size,
+ frag_size,
+ !is_lastframe_inseq);
+
+ if (fcfrag_head) {
+ next_tvb = tvb_new_chain(tvb, fcfrag_head->tvb_data);
+
+ /* Add the defragmented data to the data source list. */
+ add_new_data_source(pinfo, next_tvb, "Reassembled FC");
+
+ hidden_item = proto_tree_add_boolean (fc_tree, hf_fc_reassembled,
+ tvb, offset+9, 1, 1);
+ proto_item_set_hidden(hidden_item);
+ }
+ else {
+ hidden_item = proto_tree_add_boolean (fc_tree, hf_fc_reassembled,
+ tvb, offset+9, 1, 0);
+ proto_item_set_hidden(hidden_item);
+ next_tvb = tvb_new_subset_remaining (tvb, next_offset);
+ call_data_dissector(next_tvb, pinfo, tree);
+ return;
+ }
+ }
+ } else {
+ hidden_item = proto_tree_add_boolean (fc_tree, hf_fc_reassembled,
+ tvb, offset+9, 1, 0);
+ proto_item_set_hidden(hidden_item);
+ next_tvb = tvb_new_subset_remaining (tvb, next_offset);
+ }
+
+ if ((ftype != FC_FTYPE_LINKCTL) && (ftype != FC_FTYPE_BLS)) {
+ /* If relative offset is used, only dissect the pdu with
+ * offset 0 (param) */
+ if( (fchdr->fctl&FC_FCTL_REL_OFFSET) && param ){
+ call_data_dissector(next_tvb, pinfo, tree);
+ } else {
+ if (!dissector_try_uint_new (fcftype_dissector_table, ftype,
+ next_tvb, pinfo, tree, FALSE, fchdr)) {
+ call_data_dissector(next_tvb, pinfo, tree);
+ }
+ }
+ } else if (ftype == FC_FTYPE_BLS) {
+ if ((fchdr->r_ctl & 0x0F) == FC_BLS_BAACC) {
+ dissect_fc_ba_acc (next_tvb, pinfo, tree);
+ } else if ((fchdr->r_ctl & 0x0F) == FC_BLS_BARJT) {
+ dissect_fc_ba_rjt (next_tvb, pinfo, tree);
+ } else if ((fchdr->r_ctl & 0x0F) == FC_BLS_ABTS) {
+ col_set_str(pinfo->cinfo, COL_PROTOCOL, "BLS");
+ col_set_str(pinfo->cinfo, COL_INFO, "ABTS");
+ }
+ }
+
+ /* Lun is only populated by subdissectors, and subsequent packets assume the same lun.
+ The only way that consistently works is to save the lun on the first pass (with OXID as
+ key) when packets are guaranteed to be parsed consecutively */
+
+ /* Set up LUN data */
+ if (!pinfo->fd->visited) {
+ wmem_tree_insert32(fc_conv_data->luns, fchdr->oxid, GUINT_TO_POINTER((guint)fchdr->lun));
+ }
+
+ exchange_key = ((fchdr->oxid & 0xFFFF) | ((fchdr->lun << 16) & 0xFFFF0000));
+
+ /* set up the exchange data */
+ /* XXX we should come up with a way to handle when the 16bit oxid wraps
+ * so that large traces will work
+ */
+ fc_ex=(fc_exchange_t*)wmem_tree_lookup32(fc_conv_data->exchanges, exchange_key);
+ if(!fc_ex){
+ fc_ex=wmem_new(wmem_file_scope(), fc_exchange_t);
+ fc_ex->first_exchange_frame=0;
+ fc_ex->last_exchange_frame=0;
+ fc_ex->fc_time=pinfo->abs_ts;
+
+ wmem_tree_insert32(fc_conv_data->exchanges, exchange_key, fc_ex);
+ }
+
+ fchdr->fc_ex = fc_ex;
+
+ /* XXX: The ACK_1 frames (and other LINK_CONTROL frames) should
+ * probably be ignored (or treated specially) for SRT purposes,
+ * and not used to change the first exchange frame or start time
+ * of an exchange.
+ */
+
+ /* populate the exchange struct */
+ if(!pinfo->fd->visited){
+ if(fchdr->fctl&FC_FCTL_EXCHANGE_FIRST){
+ fc_ex->first_exchange_frame=pinfo->num;
+ fc_ex->fc_time = pinfo->abs_ts;
+ }
+ if(fchdr->fctl&FC_FCTL_EXCHANGE_LAST){
+ fc_ex->last_exchange_frame=pinfo->num;
+ }
+ }
+
+ /* put some nice exchange data in the tree */
+ if(!(fchdr->fctl&FC_FCTL_EXCHANGE_FIRST)){
+ proto_item *it;
+ it=proto_tree_add_uint(fc_tree, hf_fc_exchange_first_frame, tvb, 0, 0, fc_ex->first_exchange_frame);
+ proto_item_set_generated(it);
+ if(fchdr->fctl&FC_FCTL_EXCHANGE_LAST){
+ nstime_t delta_ts;
+ nstime_delta(&delta_ts, &pinfo->abs_ts, &fc_ex->fc_time);
+ it=proto_tree_add_time(ti, hf_fc_time, tvb, 0, 0, &delta_ts);
+ proto_item_set_generated(it);
+ }
+ }
+ if(!(fchdr->fctl&FC_FCTL_EXCHANGE_LAST)){
+ proto_item *it;
+ it=proto_tree_add_uint(fc_tree, hf_fc_exchange_last_frame, tvb, 0, 0, fc_ex->last_exchange_frame);
+ proto_item_set_generated(it);
+ }
+
+ tap_queue_packet(fc_tap, pinfo, fchdr);
+}
+
+static int
+dissect_fc (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void* data)
+{
+ fc_data_t* fc_data = (fc_data_t*)data;
+
+ if (!fc_data)
+ return 0;
+
+ dissect_fc_helper (tvb, pinfo, tree, FALSE, fc_data);
+ return tvb_captured_length(tvb);
+}
+
+static int
+dissect_fc_wtap (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void* data _U_)
+{
+ fc_data_t fc_data;
+
+ fc_data.ethertype = ETHERTYPE_UNK;
+ fc_data.sof_eof = 0;
+
+ dissect_fc_helper (tvb, pinfo, tree, FALSE, &fc_data);
+ return tvb_captured_length(tvb);
+}
+
+static int
+dissect_fc_ifcp (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void* data)
+{
+ fc_data_t* fc_data = (fc_data_t*)data;
+
+ if (!fc_data)
+ return 0;
+
+ dissect_fc_helper (tvb, pinfo, tree, TRUE, fc_data);
+ return tvb_captured_length(tvb);
+}
+
+static int
+dissect_fcsof(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void* data _U_) {
+
+ proto_item *it;
+ proto_tree *fcsof_tree;
+ tvbuff_t *next_tvb;
+ guint32 sof;
+ guint32 crc_computed;
+ guint32 eof;
+ const gint FCSOF_TRAILER_LEN = 8;
+ const gint FCSOF_HEADER_LEN = 4;
+ gint crc_offset = tvb_reported_length(tvb) - FCSOF_TRAILER_LEN;
+ gint eof_offset = crc_offset + 4;
+ gint sof_offset = 0;
+ gint frame_len_for_checksum;
+ fc_data_t fc_data;
+
+ col_set_str(pinfo->cinfo, COL_PROTOCOL, "FC");
+
+ /* Get SOF */
+ sof = tvb_get_ntohl(tvb, 0);
+
+ /* GET Computed CRC */
+ frame_len_for_checksum = crc_offset - FCSOF_HEADER_LEN;
+ crc_computed = crc32_802_tvb(tvb_new_subset_length(tvb, 4, frame_len_for_checksum), frame_len_for_checksum);
+
+ /* Get EOF */
+ eof = tvb_get_ntohl(tvb, eof_offset);
+
+ it = proto_tree_add_protocol_format(tree, proto_fcsof, tvb, 0,
+ 4, "Fibre Channel Delimiter: SOF: %s EOF: %s",
+ val_to_str(sof, fc_sof_vals, "0x%x"),
+ val_to_str(eof, fc_eof_vals, "0x%x"));
+
+ fcsof_tree = proto_item_add_subtree(it, ett_fcsof);
+
+ proto_tree_add_uint(fcsof_tree, hf_fcsof, tvb, sof_offset, 4, sof);
+
+ proto_tree_add_checksum(fcsof_tree, tvb, crc_offset, hf_fccrc, hf_fccrc_status, &ei_fccrc, pinfo, crc_computed, ENC_BIG_ENDIAN, PROTO_CHECKSUM_VERIFY);
+
+ proto_tree_add_uint(fcsof_tree, hf_fceof, tvb, eof_offset, 4, eof);
+
+ next_tvb = tvb_new_subset_length(tvb, 4, crc_offset-4);
+
+ fc_data.ethertype = ETHERTYPE_UNK;
+ fc_data.sof_eof = 0;
+ if (sof == FC_SOFI2 || sof == FC_SOFI3) {
+ fc_data.sof_eof = FC_DATA_SOF_FIRST_FRAME;
+ } else if (sof == FC_SOFF) {
+ fc_data.sof_eof = FC_DATA_SOF_SOFF;
+ }
+
+ if (eof == EOFT_POS || eof == EOFT_NEG) {
+ fc_data.sof_eof |= FC_DATA_EOF_LAST_FRAME;
+ } else if (eof == EOFDTI_NEG || eof == EOFDTI_POS) {
+ fc_data.sof_eof |= FC_DATA_EOF_INVALID;
+ }
+
+ /* Call FC dissector */
+ call_dissector_with_data(fc_handle, next_tvb, pinfo, tree, &fc_data);
+ return tvb_captured_length(tvb);
+}
+
+/* Register the protocol with Wireshark */
+
+/* this format is require because a script is used to build the C function
+ that calls all the protocol registration.
+*/
+
+void
+proto_register_fc(void)
+{
+
+/* Setup list of header fields See Section 1.6.1 for details*/
+ static hf_register_info hf[] = {
+ { &hf_fc_rctl,
+ { "R_CTL", "fc.r_ctl", FT_UINT8, BASE_HEX, NULL, 0x0,
+ NULL, HFILL }},
+ { &hf_fc_ftype,
+ {"Frame type", "fc.ftype", FT_UINT8, BASE_HEX, VALS(fc_ftype_vals),
+ 0x0, "Derived Type", HFILL}},
+ { &hf_fc_did,
+ { "Dest Addr", "fc.d_id", FT_BYTES, SEP_DOT, NULL, 0x0,
+ "Destination Address", HFILL}},
+ { &hf_fc_csctl,
+ {"CS_CTL", "fc.cs_ctl", FT_UINT8, BASE_HEX, NULL, 0x0,
+ NULL, HFILL}},
+ { &hf_fc_sid,
+ {"Src Addr", "fc.s_id", FT_BYTES, SEP_DOT, NULL, 0x0,
+ "Source Address", HFILL}},
+ { &hf_fc_id,
+ {"Addr", "fc.id", FT_BYTES, SEP_DOT, NULL, 0x0,
+ "Source or Destination Address", HFILL}},
+ { &hf_fc_type,
+ {"Type", "fc.type", FT_UINT8, BASE_HEX, VALS (fc_fc4_val), 0x0,
+ NULL, HFILL}},
+ { &hf_fc_fctl,
+ {"F_CTL", "fc.f_ctl", FT_UINT24, BASE_HEX, NULL, 0x0, NULL, HFILL}},
+ { &hf_fc_seqid,
+ {"SEQ_ID", "fc.seq_id", FT_UINT8, BASE_HEX, NULL, 0x0,
+ "Sequence ID", HFILL}},
+ { &hf_fc_dfctl,
+ {"DF_CTL", "fc.df_ctl", FT_UINT8, BASE_HEX, NULL, 0x0, NULL, HFILL}},
+ { &hf_fc_seqcnt,
+ {"SEQ_CNT", "fc.seq_cnt", FT_UINT16, BASE_DEC, NULL, 0x0,
+ "Sequence Count", HFILL}},
+ { &hf_fc_oxid,
+ {"OX_ID", "fc.ox_id", FT_UINT16, BASE_HEX, NULL, 0x0, "Originator ID",
+ HFILL}},
+ { &hf_fc_rxid,
+ {"RX_ID", "fc.rx_id", FT_UINT16, BASE_HEX, NULL, 0x0, "Receiver ID",
+ HFILL}},
+ { &hf_fc_param,
+ {"Parameter", "fc.parameter", FT_UINT32, BASE_HEX, NULL, 0x0, NULL,
+ HFILL}},
+
+ { &hf_fc_reassembled,
+ {"Reassembled Frame", "fc.reassembled", FT_BOOLEAN, BASE_NONE, NULL,
+ 0x0, NULL, HFILL}},
+ { &hf_fc_nh_da,
+ {"Network DA", "fc.nethdr.da", FT_FCWWN, BASE_NONE, NULL,
+ 0x0, NULL, HFILL}},
+ { &hf_fc_nh_sa,
+ {"Network SA", "fc.nethdr.sa", FT_FCWWN, BASE_NONE, NULL,
+ 0x0, NULL, HFILL}},
+
+ /* Basic Link Svc field definitions */
+ { &hf_fc_bls_seqid_vld,
+ {"SEQID Valid", "fc.bls_seqidvld", FT_UINT8, BASE_HEX,
+ VALS (fc_bls_seqid_val), 0x0, NULL, HFILL}},
+ { &hf_fc_bls_lastvld_seqid,
+ {"Last Valid SEQID", "fc.bls_lastseqid", FT_UINT8, BASE_HEX, NULL,
+ 0x0, NULL, HFILL}},
+ { &hf_fc_bls_oxid,
+ {"OXID", "fc.bls_oxid", FT_UINT16, BASE_HEX, NULL, 0x0, NULL, HFILL}},
+ { &hf_fc_bls_rxid,
+ {"RXID", "fc.bls_rxid", FT_UINT16, BASE_HEX, NULL, 0x0, NULL, HFILL}},
+ { &hf_fc_bls_lowseqcnt,
+ {"Low SEQCNT", "fc.bls_lseqcnt", FT_UINT16, BASE_HEX, NULL, 0x0, NULL,
+ HFILL}},
+ { &hf_fc_bls_hiseqcnt,
+ {"High SEQCNT", "fc.bls_hseqcnt", FT_UINT16, BASE_HEX, NULL, 0x0, NULL,
+ HFILL}},
+ { &hf_fc_bls_rjtcode,
+ {"Reason", "fc.bls_reason", FT_UINT8, BASE_HEX, VALS(fc_bls_barjt_val),
+ 0x0, NULL, HFILL}},
+ { &hf_fc_bls_rjtdetail,
+ {"Reason Explanation", "fc.bls_rjtdetail", FT_UINT8, BASE_HEX,
+ VALS (fc_bls_barjt_det_val), 0x0, NULL, HFILL}},
+ { &hf_fc_bls_vendor,
+ {"Vendor Unique Reason", "fc.bls_vnduniq", FT_UINT8, BASE_HEX, NULL,
+ 0x0, NULL, HFILL}},
+ { &hf_fc_fctl_exchange_responder,
+ {"ExgRpd", "fc.fctl.exchange_responder", FT_BOOLEAN, 24, TFS(&tfs_fc_fctl_exchange_responder),
+ FC_FCTL_EXCHANGE_RESPONDER, "Exchange Responder?", HFILL}},
+ { &hf_fc_fctl_seq_recipient,
+ {"SeqRec", "fc.fctl.seq_recipient", FT_BOOLEAN, 24, TFS(&tfs_fc_fctl_seq_recipient),
+ FC_FCTL_SEQ_RECIPIENT, "Seq Recipient?", HFILL}},
+ { &hf_fc_fctl_exchange_first,
+ {"ExgFst", "fc.fctl.exchange_first", FT_BOOLEAN, 24, TFS(&tfs_fc_fctl_exchange_first),
+ FC_FCTL_EXCHANGE_FIRST, "First Exchange?", HFILL}},
+ { &hf_fc_fctl_exchange_last,
+ {"ExgLst", "fc.fctl.exchange_last", FT_BOOLEAN, 24, TFS(&tfs_fc_fctl_exchange_last),
+ FC_FCTL_EXCHANGE_LAST, "Last Exchange?", HFILL}},
+ { &hf_fc_fctl_seq_last,
+ {"SeqLst", "fc.fctl.seq_last", FT_BOOLEAN, 24, TFS(&tfs_fc_fctl_seq_last),
+ FC_FCTL_SEQ_LAST, "Last Sequence?", HFILL}},
+ { &hf_fc_fctl_priority,
+ {"Pri", "fc.fctl.priority", FT_BOOLEAN, 24, TFS(&tfs_fc_fctl_priority),
+ FC_FCTL_PRIORITY, "Priority", HFILL}},
+ { &hf_fc_fctl_transfer_seq_initiative,
+ {"TSI", "fc.fctl.transfer_seq_initiative", FT_BOOLEAN, 24, TFS(&tfs_fc_fctl_transfer_seq_initiative),
+ FC_FCTL_TRANSFER_SEQ_INITIATIVE, "Transfer Seq Initiative", HFILL}},
+ { &hf_fc_fctl_rexmitted_seq,
+ {"RetSeq", "fc.fctl.rexmitted_seq", FT_BOOLEAN, 24, TFS(&tfs_fc_fctl_rexmitted_seq),
+ FC_FCTL_REXMITTED_SEQ, "Retransmitted Sequence", HFILL}},
+ { &hf_fc_fctl_rel_offset,
+ {"RelOff", "fc.fctl.rel_offset", FT_BOOLEAN, 24, TFS(&tfs_fc_fctl_rel_offset),
+ FC_FCTL_REL_OFFSET, "rel offset", HFILL}},
+ { &hf_fc_fctl_last_data_frame,
+ {"LDF", "fc.fctl.last_data_frame", FT_UINT24, BASE_HEX, VALS(last_data_frame_vals),
+ FC_FCTL_LAST_DATA_FRAME_MASK, "Last Data Frame?", HFILL}},
+ { &hf_fc_fctl_ack_0_1,
+ {"A01", "fc.fctl.ack_0_1", FT_UINT24, BASE_HEX, VALS(ack_0_1_vals),
+ FC_FCTL_ACK_0_1_MASK, "Ack 0/1 value", HFILL}},
+ { &hf_fc_fctl_abts_ack,
+ {"AA", "fc.fctl.abts_ack", FT_UINT24, BASE_HEX, VALS(abts_ack_vals),
+ FC_FCTL_ABTS_MASK, "ABTS ACK values", HFILL}},
+#if 0
+ { &hf_fc_fctl_abts_not_ack,
+ {"AnA", "fc.fctl.abts_not_ack", FT_UINT24, BASE_HEX, VALS(abts_not_ack_vals),
+ FC_FCTL_ABTS_MASK, "ABTS not ACK vals", HFILL}},
+#endif
+ { &hf_fc_exchange_first_frame,
+ { "Exchange First In", "fc.exchange_first_frame", FT_FRAMENUM, BASE_NONE, NULL,
+ 0, "The first frame of this exchange is in this frame", HFILL }},
+ { &hf_fc_exchange_last_frame,
+ { "Exchange Last In", "fc.exchange_last_frame", FT_FRAMENUM, BASE_NONE, NULL,
+ 0, "The last frame of this exchange is in this frame", HFILL }},
+ { &hf_fc_time,
+ { "Time from Exchange First", "fc.time", FT_RELATIVE_TIME, BASE_NONE, NULL,
+ 0, "Time since the first frame of the Exchange", HFILL }},
+ { &hf_fc_relative_offset,
+ {"Relative Offset", "fc.relative_offset", FT_UINT32, BASE_DEC, NULL,
+ 0, "Relative offset of data", HFILL}},
+ { &hf_fc_vft,
+ {"VFT Header", "fc.vft", FT_UINT16, BASE_DEC, NULL,
+ 0, NULL, HFILL}},
+ { &hf_fc_vft_rctl,
+ {"R_CTL", "fc.vft.rctl", FT_UINT8, BASE_HEX, NULL,
+ 0, NULL, HFILL}},
+ { &hf_fc_vft_ver,
+ {"Version", "fc.vft.ver", FT_UINT8, BASE_DEC, NULL,
+ 0, "Version of VFT header", HFILL}},
+ { &hf_fc_vft_type,
+ {"Type", "fc.vft.type", FT_UINT8, BASE_DEC, NULL,
+ 0, "Type of tagged frame", HFILL}},
+ { &hf_fc_vft_pri,
+ {"Priority", "fc.vft.pri", FT_UINT8, BASE_DEC, NULL,
+ 0, "QoS Priority", HFILL}},
+ { &hf_fc_vft_vf_id,
+ {"VF_ID", "fc.vft.vf_id", FT_UINT16, BASE_DEC, NULL,
+ 0, "Virtual Fabric ID", HFILL}},
+ { &hf_fc_vft_hop_ct,
+ {"HopCT", "fc.vft.hop_ct", FT_UINT8, BASE_DEC, NULL,
+ 0, "Hop Count", HFILL}},
+ };
+
+ /* Setup protocol subtree array */
+ static gint *ett[] = {
+ &ett_fc,
+ &ett_fcbls,
+ &ett_fc_vft,
+ &ett_fctl
+ };
+
+ static ei_register_info ei[] = {
+ { &ei_fccrc,
+ { "fc.crc.bad", PI_CHECKSUM, PI_ERROR, "Bad checksum", EXPFILL }},
+ { &ei_short_hdr,
+ { "fc.short_hdr", PI_MALFORMED, PI_ERROR,
+ "Packet length is shorter than the required header", EXPFILL }},
+#if 0
+ { &ei_frag_size,
+ { "fc.frag_size", PI_MALFORMED, PI_ERROR,
+ "Invalid fragment size", EXPFILL }}
+#endif
+ };
+
+ module_t *fc_module;
+ expert_module_t* expert_fc;
+
+ /* FC SOF */
+
+ static hf_register_info sof_hf[] = {
+ { &hf_fcsof,
+ { "SOF", "fc.sof", FT_UINT32, BASE_HEX, VALS(fc_sof_vals), 0,
+ NULL, HFILL }},
+ { &hf_fceof,
+ { "EOF", "fc.eof", FT_UINT32, BASE_HEX, VALS(fc_eof_vals), 0,
+ NULL, HFILL }},
+ { &hf_fccrc,
+ { "CRC", "fc.crc", FT_UINT32, BASE_HEX, NULL, 0, NULL, HFILL }},
+ { &hf_fccrc_status,
+ { "CRC Status", "fc.crc.status", FT_UINT8, BASE_NONE, VALS(proto_checksum_vals), 0, NULL, HFILL }},
+ };
+
+ static gint *sof_ett[] = {
+ &ett_fcsof,
+ &ett_fceof,
+ &ett_fccrc
+ };
+
+
+ /* Register the protocol name and description */
+ proto_fc = proto_register_protocol ("Fibre Channel", "FC", "fc");
+ fc_handle = register_dissector ("fc", dissect_fc, proto_fc);
+ register_dissector ("fc_ifcp", dissect_fc_ifcp, proto_fc);
+ fc_tap = register_tap("fc");
+
+ /* Required function calls to register the header fields and subtrees used */
+ proto_register_field_array(proto_fc, hf, array_length(hf));
+ proto_register_subtree_array(ett, array_length(ett));
+ expert_fc = expert_register_protocol(proto_fc);
+ expert_register_field_array(expert_fc, ei, array_length(ei));
+
+ /* subdissectors called through this table will find the fchdr structure
+ * through data parameter of dissector
+ */
+ fcftype_dissector_table = register_dissector_table ("fc.ftype",
+ "FC Frame Type",
+ proto_fc, FT_UINT8, BASE_HEX);
+
+ /* Register preferences */
+ fc_module = prefs_register_protocol (proto_fc, NULL);
+ prefs_register_bool_preference (fc_module,
+ "reassemble",
+ "Reassemble multi-frame sequences",
+ "If enabled, reassembly of multi-frame "
+ "sequences is done",
+ &fc_reassemble);
+ prefs_register_uint_preference (fc_module,
+ "max_frame_size", "Max FC Frame Size",
+ "This is the size of non-last frames in a "
+ "multi-frame sequence", 10,
+ &fc_max_frame_size);
+
+ fcseq_req_hash = wmem_map_new_autoreset(wmem_epan_scope(), wmem_file_scope(), fcseq_hash, fcseq_equal);
+
+ reassembly_table_register(&fc_reassembly_table,
+ &addresses_reassembly_table_functions);
+
+
+ /* Register FC SOF/EOF */
+ proto_fcsof = proto_register_protocol("Fibre Channel Delimiters", "FCSoF", "fcsof");
+
+ proto_register_field_array(proto_fcsof, sof_hf, array_length(sof_hf));
+ proto_register_subtree_array(sof_ett, array_length(sof_ett));
+
+ fcsof_handle = register_dissector("fcsof", dissect_fcsof, proto_fcsof);
+
+ register_conversation_table(proto_fc, TRUE, fc_conversation_packet, fc_endpoint_packet);
+ register_srt_table(proto_fc, NULL, 1, fcstat_packet, fcstat_init, NULL);
+}
+
+
+/* If this dissector uses sub-dissector registration add a registration routine.
+ This format is required because a script is used to find these routines and
+ create the code that calls these routines.
+*/
+void
+proto_reg_handoff_fc (void)
+{
+ dissector_add_uint("wtap_encap", WTAP_ENCAP_FIBRE_CHANNEL_FC2,
+ create_dissector_handle(dissect_fc_wtap, proto_fc));
+
+ dissector_add_uint("wtap_encap", WTAP_ENCAP_FIBRE_CHANNEL_FC2_WITH_FRAME_DELIMS, fcsof_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:
+ */