summaryrefslogtreecommitdiffstats
path: root/epan/dissectors/packet-fcels.c
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-10 20:34:10 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-10 20:34:10 +0000
commite4ba6dbc3f1e76890b22773807ea37fe8fa2b1bc (patch)
tree68cb5ef9081156392f1dd62a00c6ccc1451b93df /epan/dissectors/packet-fcels.c
parentInitial commit. (diff)
downloadwireshark-e4ba6dbc3f1e76890b22773807ea37fe8fa2b1bc.tar.xz
wireshark-e4ba6dbc3f1e76890b22773807ea37fe8fa2b1bc.zip
Adding upstream version 4.2.2.upstream/4.2.2
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'epan/dissectors/packet-fcels.c')
-rw-r--r--epan/dissectors/packet-fcels.c2621
1 files changed, 2621 insertions, 0 deletions
diff --git a/epan/dissectors/packet-fcels.c b/epan/dissectors/packet-fcels.c
new file mode 100644
index 00000000..cde67885
--- /dev/null
+++ b/epan/dissectors/packet-fcels.c
@@ -0,0 +1,2621 @@
+/* packet-fcels.c
+ * Routines for FC Extended Link Services
+ * Copyright 2001, Dinesh G Dutt <ddutt@cisco.com>
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+/*
+ * TODO Still (Complete compliance with FC-MI):
+ * - Decode RNID, RLIR
+ */
+
+#include "config.h"
+
+#include <epan/packet.h>
+#include <epan/to_str.h>
+#include <epan/expert.h>
+#include "packet-fc.h"
+#include "packet-fcels.h"
+
+void proto_register_fcels(void);
+void proto_reg_handoff_fcels(void);
+
+static dissector_handle_t els_handle;
+
+#define FC_ELS_RPLY 0
+#define FC_ELS_REQ 1
+
+/* Initialize the protocol and registered fields */
+static int proto_fcels = -1;
+static int hf_fcels_opcode = -1;
+static int hf_fcels_rjtcode = -1;
+static int hf_fcels_rjtdetcode = -1;
+static int hf_fcels_vnduniq = -1;
+static int hf_fcels_b2b = -1;
+static int hf_fcels_cmnfeatures = -1;
+static int hf_fcels_bbscnum = -1;
+static int hf_fcels_rcvsize = -1;
+static int hf_fcels_maxconseq = -1;
+static int hf_fcels_reloffset = -1;
+static int hf_fcels_edtov = -1;
+static int hf_fcels_npname = -1;
+static int hf_fcels_fnname = -1;
+#if 0
+static int hf_fcels_cls1param = -1;
+static int hf_fcels_cls2param = -1;
+static int hf_fcels_cls3param = -1;
+static int hf_fcels_cls4param = -1;
+#endif
+static int hf_fcels_vendorvers = -1;
+static int hf_fcels_svcavail = -1;
+static int hf_fcels_clsflags = -1;
+static int hf_fcels_clsrcvsize = -1;
+static int hf_fcels_conseq = -1;
+static int hf_fcels_e2e = -1;
+static int hf_fcels_openseq = -1;
+static int hf_fcels_nportid = -1;
+static int hf_fcels_oxid = -1;
+static int hf_fcels_rxid = -1;
+static int hf_fcels_recovqual = -1;
+static int hf_fcels_fabricaddr = -1;
+static int hf_fcels_fabricpname = -1;
+static int hf_fcels_failedrcvr = -1;
+static int hf_fcels_flacompliance = -1;
+static int hf_fcels_loopstate = -1;
+static int hf_fcels_publicloop_bmap = -1;
+static int hf_fcels_pvtloop_bmap = -1;
+static int hf_fcels_alpa_map = -1;
+static int hf_fcels_scrregn = -1;
+static int hf_fcels_farp_matchcodept = -1;
+static int hf_fcels_farp_respaction = -1;
+static int hf_fcels_resportid = -1;
+static int hf_fcels_respname = -1;
+static int hf_fcels_respnname = -1;
+static int hf_fcels_reqipaddr = -1;
+static int hf_fcels_respipaddr = -1;
+static int hf_fcels_hardaddr = -1;
+static int hf_fcels_rps_flag = -1;
+static int hf_fcels_rps_portnum = -1;
+static int hf_fcels_rps_portstatus = -1;
+static int hf_fcels_rnft_fc4type = -1;
+static int hf_fcels_rscn_evqual = -1;
+static int hf_fcels_rscn_addrfmt = -1;
+static int hf_fcels_rscn_domain = -1;
+static int hf_fcels_rscn_area = -1;
+static int hf_fcels_rscn_port = -1;
+static int hf_fcels_rec_fc4 = -1;
+static int hf_fcels_estat = -1;
+static int hf_fcels_estat_resp = -1;
+static int hf_fcels_estat_seq_init = -1;
+static int hf_fcels_estat_compl = -1;
+static int hf_fcels_nodeidfmt = -1;
+static int hf_fcels_spidlen = -1;
+static int hf_fcels_vendoruniq = -1;
+static int hf_fcels_vendorsp = -1;
+static int hf_fcels_asstype = -1;
+static int hf_fcels_physport = -1;
+static int hf_fcels_attnodes = -1;
+static int hf_fcels_nodemgmt = -1;
+static int hf_fcels_ipvers = -1;
+static int hf_fcels_tcpport = -1;
+static int hf_fcels_ip = -1;
+static int hf_fcels_cbind_liveness = -1;
+static int hf_fcels_cbind_addr_mode = -1;
+static int hf_fcels_cbind_ifcp_version = -1;
+static int hf_fcels_cbind_userinfo = -1;
+static int hf_fcels_cbind_snpname = -1;
+static int hf_fcels_cbind_dnpname = -1;
+static int hf_fcels_cbind_status = -1;
+static int hf_fcels_chandle = -1;
+static int hf_fcels_unbind_status = -1;
+static int hf_fcels_cmn_cios = -1;
+static int hf_fcels_cmn_rro = -1;
+static int hf_fcels_cmn_vvv = -1;
+static int hf_fcels_cmn_b2b = -1;
+static int hf_fcels_cmn_e_d_tov = -1;
+static int hf_fcels_cmn_simplex = -1;
+static int hf_fcels_cmn_multicast = -1;
+static int hf_fcels_cmn_broadcast = -1;
+static int hf_fcels_cmn_security = -1;
+static int hf_fcels_cmn_clk = -1;
+static int hf_fcels_cmn_dhd = -1;
+static int hf_fcels_cmn_seqcnt = -1;
+static int hf_fcels_cmn_payload = -1;
+static int hf_fcels_cls_cns = -1;
+static int hf_fcels_cls_sdr = -1;
+static int hf_fcels_cls_prio = -1;
+static int hf_fcels_cls_nzctl = -1;
+static int hf_fcels_initctl = -1;
+static int hf_fcels_initctl_initial_pa = -1;
+static int hf_fcels_initctl_ack0 = -1;
+static int hf_fcels_initctl_ackgaa = -1;
+static int hf_fcels_initctl_sync = -1;
+static int hf_fcels_rcptctl = -1;
+static int hf_fcels_rcptctl_ack0 = -1;
+static int hf_fcels_rcptctl_interlock = -1;
+static int hf_fcels_rcptctl_policy = -1;
+static int hf_fcels_rcptctl_category = -1;
+static int hf_fcels_rcptctl_sync = -1;
+static int hf_fcels_fcpflags = -1;
+static int hf_fcels_fcpflags_trireq = -1;
+static int hf_fcels_fcpflags_trirep = -1;
+static int hf_fcels_fcpflags_retry = -1;
+static int hf_fcels_fcpflags_ccomp = -1;
+static int hf_fcels_fcpflags_datao = -1;
+static int hf_fcels_fcpflags_initiator = -1;
+static int hf_fcels_fcpflags_target = -1;
+static int hf_fcels_fcpflags_rdxr = -1;
+static int hf_fcels_fcpflags_wrxr = -1;
+static int hf_fcels_prliloflags = -1;
+static int hf_fcels_tprloflags_opav = -1;
+static int hf_fcels_tprloflags_rpav = -1;
+static int hf_fcels_tprloflags_npv = -1;
+static int hf_fcels_tprloflags_gprlo = -1;
+static int hf_fcels_speedflags = -1;
+static int hf_fcels_speedflags_1gb = -1;
+static int hf_fcels_speedflags_2gb = -1;
+static int hf_fcels_speedflags_4gb = -1;
+static int hf_fcels_speedflags_10gb = -1;
+static int hf_fcels_prliloflags_opav = -1;
+static int hf_fcels_prliloflags_ipe = -1;
+static int hf_fcels_prliloflags_eip = -1;
+/* Generated from convert_proto_tree_add_text.pl */
+static int hf_fcels_rnft_index_of_first_rec_in_list = -1;
+static int hf_fcels_lip_f7_received_count = -1;
+static int hf_fcels_recovery_qualifier_status = -1;
+static int hf_fcels_rpl_port_identifier = -1;
+static int hf_fcels_rpl_index_of_i_port_block = -1;
+static int hf_fcels_lip_f7_initiated_count = -1;
+static int hf_fcels_srl_fl_port_addr = -1;
+static int hf_fcels_loss_of_signal_count = -1;
+static int hf_fcels_lirr_regn_function = -1;
+static int hf_fcels_rscn_page_len = -1;
+static int hf_fcels_prlilo_service_parameter_response = -1;
+static int hf_fcels_prlilo_type_code_extension = -1;
+static int hf_fcels_rnft_list_length = -1;
+static int hf_fcels_rpl_index = -1;
+static int hf_fcels_rpl_physical_port = -1;
+static int hf_fcels_prlilo_originator_pa = -1;
+static int hf_fcels_rpl_list_length = -1;
+static int hf_fcels_common_identification_data_length = -1;
+static int hf_fcels_loss_of_sync_count = -1;
+static int hf_fcels_lip_reset_received_count = -1;
+static int hf_fcels_rpl_max_size = -1;
+static int hf_fcels_prlilo_response_code = -1;
+static int hf_fcels_invalid_crc_count = -1;
+static int hf_fcels_rscn_payload_len = -1;
+static int hf_fcels_link_failure_count = -1;
+static int hf_fcels_prlilo_3rd_party_n_port_id = -1;
+static int hf_fcels_lip_al_ps = -1;
+static int hf_fcels_prlilo_type = -1;
+static int hf_fcels_lirr_regn_format = -1;
+static int hf_fcels_srl_flag = -1;
+static int hf_fcels_prlilo_page_length = -1;
+static int hf_fcels_rpl_payload_length = -1;
+static int hf_fcels_rpsc_port_oper_speed = -1;
+static int hf_fcels_lip_reset_initiated_count = -1;
+static int hf_fcels_l_port_status = -1;
+static int hf_fcels_primitive_seq_protocol_err = -1;
+static int hf_fcels_rnft_max_size = -1;
+static int hf_fcels_lip_f8_received_count = -1;
+static int hf_fcels_rnft_index = -1;
+static int hf_fcels_rnft_payload_len = -1;
+static int hf_fcels_prlilo_payload_length = -1;
+static int hf_fcels_prlilo_responder_pa = -1;
+static int hf_fcels_rpsc_number_of_entries = -1;
+static int hf_fcels_prlilo_3rd_party_originator_pa = -1;
+static int hf_fcels_invalid_xmission_word = -1;
+static int hf_fcels_rnft_fc4_qualifier = -1;
+static int hf_fcels_lip_f8_initiated_count = -1;
+static int hf_fcels_rpl_port_name = -1;
+
+static gint ett_fcels = -1;
+static gint ett_fcels_lsrjt = -1;
+static gint ett_fcels_acc = -1;
+static gint ett_fcels_logi = -1;
+static gint ett_fcels_logi_cmnsvc = -1;
+static gint ett_fcels_logi_clssvc = -1;
+static gint ett_fcels_logo = -1;
+static gint ett_fcels_abtx = -1;
+static gint ett_fcels_rsi = -1;
+static gint ett_fcels_rrq = -1;
+static gint ett_fcels_rec = -1;
+static gint ett_fcels_prli = -1;
+static gint ett_fcels_prli_svcpg = -1;
+static gint ett_fcels_adisc = -1;
+static gint ett_fcels_farp = -1;
+static gint ett_fcels_rps = -1;
+static gint ett_fcels_rpl = -1;
+static gint ett_fcels_rplpb = -1;
+static gint ett_fcels_fan = -1;
+static gint ett_fcels_rscn = -1;
+static gint ett_fcels_rscn_rec = -1;
+static gint ett_fcels_estat = -1;
+static gint ett_fcels_scr = -1;
+static gint ett_fcels_rnft = -1;
+static gint ett_fcels_rnft_fc4 = -1;
+static gint ett_fcels_lsts = -1;
+static gint ett_fcels_rnid = -1;
+static gint ett_fcels_rlir = -1;
+static gint ett_fcels_lirr = -1;
+static gint ett_fcels_srl = -1;
+static gint ett_fcels_rpsc = -1;
+static gint ett_fcels_cbind = -1;
+static gint ett_fcels_cmnfeatures = -1;
+static gint ett_fcels_clsflags = -1;
+static gint ett_fcels_initctl = -1;
+static gint ett_fcels_rcptctl = -1;
+static gint ett_fcels_fcpflags = -1;
+static gint ett_fcels_prliloflags = -1;
+static gint ett_fcels_speedflags = -1;
+
+static expert_field ei_fcels_src_unknown = EI_INIT;
+static expert_field ei_fcels_dst_unknown = EI_INIT;
+static expert_field ei_fcels_no_record_of_els_req = EI_INIT;
+static expert_field ei_fcels_no_record_of_exchange = EI_INIT;
+
+static int * const hf_fcels_estat_fields[] = {
+ &hf_fcels_estat_resp,
+ &hf_fcels_estat_seq_init,
+ &hf_fcels_estat_compl,
+ NULL
+};
+
+static const true_false_string tfs_fcels_estat_resp = {
+ "Responding to Exchange",
+ "Originator of Exchange"
+};
+
+static const true_false_string tfs_fcels_estat_seq_init = {
+ "Seq Initiative held by REC responder",
+ "Seq Initiative not held by REC responder"
+};
+
+
+#define FC_ESB_ST_RESP (1U << 31) /* responder to exchange */
+#define FC_ESB_ST_SEQ_INIT (1U << 30) /* holds sequence initiative */
+#define FC_ESB_ST_COMPLETE (1U << 29) /* exchange is complete */
+
+static const value_string fc_els_proto_val[] = {
+ {FC_ELS_LSRJT , "LS_RJT"},
+ {FC_ELS_ACC , "ACC"},
+ {FC_ELS_PLOGI , "PLOGI"},
+ {FC_ELS_FLOGI , "FLOGI"},
+ {FC_ELS_LOGOUT , "LOGO"},
+ {FC_ELS_ABTX , "ABTX"},
+ {FC_ELS_RSI , "RSI"},
+ {FC_ELS_RTV , "RTV"},
+ {FC_ELS_RLS , "RLS"},
+ {FC_ELS_ECHO , "ECHO"},
+ {FC_ELS_TEST , "TEST"},
+ {FC_ELS_RRQ , "RRQ"},
+ {FC_ELS_REC , "REC"},
+ {FC_ELS_SRR , "SRR"},
+ {FC_ELS_PRLI , "PRLI"},
+ {FC_ELS_PRLO , "PRLO"},
+ {FC_ELS_TPRLO , "TPRLO"},
+ {FC_ELS_PDISC , "PDISC"},
+ {FC_ELS_FDISC , "FDISC"},
+ {FC_ELS_ADISC , "ADISC"},
+ {FC_ELS_FARP_REQ , "FARP-REQ"},
+ {FC_ELS_FARP_RPLY , "FARP-REPLY"},
+ {FC_ELS_RPS , "RPS"},
+ {FC_ELS_RPL , "RPL"},
+ {FC_ELS_FAN , "FAN"},
+ {FC_ELS_RSCN , "RSCN"},
+ {FC_ELS_SCR , "SCR"},
+ {FC_ELS_RNFT , "RNFT"},
+ {FC_ELS_LINIT , "LINIT"},
+ {FC_ELS_LSTS , "LSTS"},
+ {FC_ELS_RNID , "RNID"},
+ {FC_ELS_RLIR , "RLIR"},
+ {FC_ELS_LIRR , "LIRR"},
+ {FC_ELS_SRL , "SRL"},
+ {FC_ELS_RPSC , "RPSC"},
+ {FC_ELS_LKA , "LKA"},
+ {FC_ELS_AUTH , "AUTH"},
+ {FC_ELS_CBIND , "CBIND"},
+ {FC_ELS_UNBIND , "UNBIND"},
+ {0, NULL}
+};
+value_string_ext fc_els_proto_val_ext = VALUE_STRING_EXT_INIT(fc_els_proto_val);
+
+/* Reject Reason Codes */
+#define FC_ELS_RJT_INVCMDCODE 0x01
+#define FC_ELS_RJT_LOGERR 0x03
+#define FC_ELS_RJT_LOGBSY 0x05
+#define FC_ELS_RJT_PROTERR 0x07
+#define FC_ELS_RJT_GENFAIL 0x09
+#define FC_ELS_RJT_CMDNOTSUPP 0x0B
+#define FC_ELS_RJT_GENFAIL2 0x0D
+#define FC_ELS_RJT_CMDINPROG 0x0E
+#define FC_ELS_RJT_FIP 0x20
+#define FC_ELS_RJT_VENDOR 0xFF
+
+static const value_string fc_els_rjt_val[] = {
+ {FC_ELS_RJT_INVCMDCODE, "Invalid Cmd Code"},
+ {FC_ELS_RJT_LOGERR , "Logical Error"},
+ {FC_ELS_RJT_LOGBSY , "Logical Busy"},
+ {FC_ELS_RJT_PROTERR , "Protocol Error"},
+ {FC_ELS_RJT_GENFAIL , "Unable to Perform Cmd"},
+ {FC_ELS_RJT_CMDNOTSUPP, "Command Not Supported"},
+ {FC_ELS_RJT_GENFAIL2 , "Unable to Perform Cmd"},
+ {FC_ELS_RJT_CMDINPROG , "Command in Progress Already"},
+ {FC_ELS_RJT_FIP , "FIP Error"},
+ {FC_ELS_RJT_VENDOR , "Vendor Unique Error"},
+ {0, NULL}
+};
+static value_string_ext fc_els_rjt_val_ext = VALUE_STRING_EXT_INIT(fc_els_rjt_val);
+
+#define FC_ELS_RJT_DET_NODET 0x00
+#define FC_ELS_RJT_DET_SVCPARM_OPT 0x01
+#define FC_ELS_RJT_DET_SVCPARM_INITCTL 0x03
+#define FC_ELS_RJT_DET_SVCPARM_RCPTCTL 0x05
+#define FC_ELS_RJT_DET_SVCPARM_RCVSZE 0x07
+#define FC_ELS_RJT_DET_SVCPARM_CSEQ 0x09
+#define FC_ELS_RJT_DET_SVCPARM_CREDIT 0x0B
+#define FC_ELS_RJT_DET_INV_PFNAME 0x0D
+#define FC_ELS_RJT_DET_INV_NFNAME 0x0E
+#define FC_ELS_RJT_DET_INV_CMNSVCPARM 0x0F
+#define FC_ELS_RJT_DET_INV_ASSOCHDR 0x11
+#define FC_ELS_RJT_DET_ASSOCHDR_REQD 0x13
+#define FC_ELS_RJT_DET_INV_OSID 0x15
+#define FC_ELS_RJT_DET_EXCHG_COMBO 0x17
+#define FC_ELS_RJT_DET_CMDINPROG 0x19
+#define FC_ELS_RJT_DET_PLOGI_REQ 0x1E
+#define FC_ELS_RJT_DET_INV_NPID 0x1F
+#define FC_ELS_RJT_DET_INV_SEQID 0x21
+#define FC_ELS_RJT_DET_INV_EXCHG 0x23
+#define FC_ELS_RJT_DET_INACTIVE_EXCHG 0x25
+#define FC_ELS_RJT_DET_RQUAL_REQD 0x27
+#define FC_ELS_RJT_DET_OORSRC 0x29
+#define FC_ELS_RJT_DET_SUPPLYFAIL 0x2A
+#define FC_ELS_RJT_DET_REQNOTSUPP 0x2C
+#define FC_ELS_RJT_DET_INV_PLEN 0x2D
+#define FC_ELS_RJT_DET_INV_ALIASID 0x30
+#define FC_ELS_RJT_DET_OORSRC_ALIASID 0x31
+#define FC_ELS_RJT_DET_INACTIVE_ALIASID 0x32
+#define FC_ELS_RJT_DET_DEACT_ALIAS_FAIL1 0x33
+#define FC_ELS_RJT_DET_DEACT_ALIAS_FAIL2 0x34
+#define FC_ELS_RJT_DET_SVCPARM_CONFLICT 0x35
+#define FC_ELS_RJT_DET_INV_ALIASTOK 0x36
+#define FC_ELS_RJT_DET_UNSUPP_ALIASTOK 0x37
+#define FC_ELS_RJT_DET_GRPFORM_FAIL 0x38
+#define FC_ELS_RJT_DET_QOSPARM_ERR 0x40
+#define FC_ELS_RJT_DET_INV_VCID 0x41
+#define FC_ELS_RJT_DET_OORSRC_C4 0x42
+#define FC_ELS_RJT_DET_INV_PNNAME 0x44
+#define FC_ELS_RJT_DET_AUTH_REQD 0x48
+#define FC_ELS_RJT_DET_NOT_NEIGHBOR 0x62
+
+static const value_string fc_els_rjt_det_val[] = {
+ {FC_ELS_RJT_DET_NODET , "No further details"},
+ {FC_ELS_RJT_DET_SVCPARM_OPT , "Svc Param - Options Error"},
+ {FC_ELS_RJT_DET_SVCPARM_INITCTL , "Svc Param - Initiator Ctl Error"},
+ {FC_ELS_RJT_DET_SVCPARM_RCPTCTL , "Svc Param - Recipient Ctl Error"},
+ {FC_ELS_RJT_DET_SVCPARM_RCVSZE , "Svc Param - Recv Size Error"},
+ {FC_ELS_RJT_DET_SVCPARM_CSEQ , "Svc Param - Concurrent Seq Error"},
+ {FC_ELS_RJT_DET_SVCPARM_CREDIT , "Svc Param - Credit Error"},
+ {FC_ELS_RJT_DET_INV_PFNAME , "Invalid N_/F_Port Name"},
+ {FC_ELS_RJT_DET_INV_NFNAME , "Invalid Node/Fabric Name"},
+ {FC_ELS_RJT_DET_INV_CMNSVCPARM , "Invalid Common Svc Param"},
+ {FC_ELS_RJT_DET_INV_ASSOCHDR , "Invalid Association Header"},
+ {FC_ELS_RJT_DET_ASSOCHDR_REQD , "Association Header Reqd"},
+ {FC_ELS_RJT_DET_INV_OSID , "Invalid Orig S_ID"},
+ {FC_ELS_RJT_DET_EXCHG_COMBO , "Invalid OXID-RXID Combo"},
+ {FC_ELS_RJT_DET_CMDINPROG , "Cmd Already in Progress"},
+ {FC_ELS_RJT_DET_PLOGI_REQ , "N_Port Login Required"},
+ {FC_ELS_RJT_DET_INV_NPID , "Invalid N_Port Id"},
+ {FC_ELS_RJT_DET_INV_SEQID , "Invalid SeqID"},
+ {FC_ELS_RJT_DET_INV_EXCHG , "Attempt to Abort Invalid Exchg"},
+ {FC_ELS_RJT_DET_INACTIVE_EXCHG , "Attempt to Abort Inactive Exchg"},
+ {FC_ELS_RJT_DET_RQUAL_REQD , "Resource Qualifier Required"},
+ {FC_ELS_RJT_DET_OORSRC , "Insufficient Resources for Login"},
+ {FC_ELS_RJT_DET_SUPPLYFAIL , "Unable to Supply Req Data"},
+ {FC_ELS_RJT_DET_REQNOTSUPP , "Command Not Supported"},
+ {FC_ELS_RJT_DET_INV_PLEN , "Invalid Payload Length"},
+ {FC_ELS_RJT_DET_INV_ALIASID , "No Alias IDs available"},
+ {FC_ELS_RJT_DET_OORSRC_ALIASID , "Alias_ID Cannot be Activated (Out of Rsrc)"},
+ {FC_ELS_RJT_DET_INACTIVE_ALIASID , "Alias_ID Cannot be Activated (Inv AID)"},
+ {FC_ELS_RJT_DET_DEACT_ALIAS_FAIL1, "Alias_ID Cannot be Deactivated"},
+ {FC_ELS_RJT_DET_DEACT_ALIAS_FAIL2, "Alias_ID Cannot be Deactivated"},
+ {FC_ELS_RJT_DET_SVCPARM_CONFLICT , "Svc Parameter Conflict"},
+ {FC_ELS_RJT_DET_INV_ALIASTOK , "Invalid Alias Token"},
+ {FC_ELS_RJT_DET_UNSUPP_ALIASTOK , "Unsupported Alias Token"},
+ {FC_ELS_RJT_DET_GRPFORM_FAIL , "Alias Grp Cannot be Formed"},
+ {FC_ELS_RJT_DET_QOSPARM_ERR , "QoS Param Error"},
+ {FC_ELS_RJT_DET_INV_VCID , "VC_ID Not Found"},
+ {FC_ELS_RJT_DET_OORSRC_C4 , "No Resources to Support Class 4 Conn"},
+ {FC_ELS_RJT_DET_INV_PNNAME , "Invalid Port/Node Name"},
+ {FC_ELS_RJT_DET_AUTH_REQD , "Authentication Required"},
+ {FC_ELS_RJT_DET_NOT_NEIGHBOR , "VN2VN_Port not in Neighbor Set"},
+ {0, NULL}
+};
+static value_string_ext fc_els_rjt_det_val_ext = VALUE_STRING_EXT_INIT(fc_els_rjt_det_val);
+
+static const value_string fc_els_flacompliance_val[] = {
+ {1, "FC-FLA Level 1"},
+ {2, "FC-FLA Level 2"},
+ {0, NULL}
+};
+
+static const value_string fc_els_loopstate_val[] = {
+ {1, "Online"},
+ {2, "Loop Failure"},
+ {3, "Initialization Failure"},
+ {4, "Initializing"},
+ {0, NULL}
+};
+
+static const value_string fc_els_scr_reg_val[] = {
+ {1, "Fabric Detected Regn"},
+ {2, "N_Port Detected Regn"},
+ {3, "Full Regn"},
+ {255, "Clear All Regn"},
+ {0, NULL}
+};
+
+static const value_string fc_els_farp_respaction_val[] = {
+ {0, "No Action"},
+ {1, "Login Using Requesting Port ID"},
+ {2, "Respond with FARP-REPLY"},
+ {3, "Login & send FARP-REPLY"},
+ {0, NULL}
+};
+
+static const value_string fc_els_portstatus_val[] = {
+ {0x01, "Link Reset Protocol in Progress"},
+ {0x02, "Loss of Synchronization"},
+ {0x04, "Loss of Signal"},
+ {0x10, "AL Connection | No Fabric"},
+ {0x14, "AL Connection | Loss of Signal"},
+ {0x18, "AL Connection | Fabric Detected"},
+ {0x1C, "AL Connection | Fabric Detected | Loss of Signal"},
+ {0x20, "Point-to-Point Connection | No Fabric"},
+ {0x24, "Point-to-Point Connection | Loss of Signal"},
+ {0x28, "Point-to-Point Connection | Fabric Detected"},
+ {0x2C, "Point-to-Point Connection | Fabric Detected | Loss of Signal"},
+ {0, NULL}
+};
+static value_string_ext fc_els_portstatus_val_ext = VALUE_STRING_EXT_INIT(fc_els_portstatus_val);
+
+static const value_string fc_els_portspeed_val[] = {
+ {0x8000, "1 Gb"},
+ {0x4000, "2 Gb"},
+ {0x2000, "4 Gb"},
+ {0x1000, "10 Gb"},
+ {0x0002, "Unknown"},
+ {0x0001, "Speed Not Estd."},
+ {0, NULL}
+};
+
+static const value_string fc_els_lirr_regfunc_val[] = {
+ {0x1, "Set Reg: Conditionally Receive"},
+ {0x2, "Set Reg: Always Receive"},
+ {0xFF, "Clear Reg"},
+ {0, NULL}
+};
+
+static const value_string fc_els_rscn_evqual_val[] = {
+ {0x00, "Event is not specified"},
+ {0x01, "Changed Name Server Object"},
+ {0x02, "Changed Port Attribute"},
+ {0x03, "Changed Service Object"},
+ {0x04, "Changed Switch Config"},
+ {0, NULL}
+};
+
+static const value_string fc_els_rscn_addrfmt_val[] = {
+ {0, "Port Addr (single N/L Port or service)"},
+ {1, "Area Addr Group (area of E/L/N Port addresses)"},
+ {2, "Domain Addr Group"},
+ {3, "Fabric Addr Group"},
+ {0, NULL}
+};
+
+static const value_string fc_els_nodeid_val[] = {
+ {0x00, "Common Identification Data Only"},
+ {0x05, "IP Specific Data"},
+ {0x08, "FCP-Specific Data"},
+ {0x20, "FC_CT Specific Data"},
+ {0x22, "SW_ILS Specific Data"},
+ {0x23, "AL Specific Data"},
+ {0x24, "SNMP Specific Data"},
+ {0xDF, "Common ID Data + General Topology Discovery Format"},
+ {0, NULL}
+};
+
+static const value_string fc_els_rnid_asstype_val[] = {
+ {0x0, "Reserved"},
+ {0x1, "Unknown"},
+ {0x2, "Other"},
+ {0x3, "Hub"},
+ {0x4, "Switch"},
+ {0x5, "Gateway"},
+ {0x6, "Converter"},
+ {0x7, "HBA"},
+ {0x9, "Storage Device"},
+ {0xA, "Host"},
+ {0xB, "Storage Subsystem"},
+ {0xE, "Storage Access Device"},
+ {0x11, "NAS Device"},
+ {0, NULL}
+};
+static value_string_ext fc_els_rnid_asstype_val_ext = VALUE_STRING_EXT_INIT(fc_els_rnid_asstype_val);
+
+static const value_string fc_els_rnid_mgmt_val[] = {
+ {0, "IP/UDP/SNMP"},
+ {1, "IP/TCP/Telnet"},
+ {2, "IP/TCP/HTTP"},
+ {3, "IP/TCP/HTTPS"},
+ {0, NULL}
+};
+
+static const value_string fc_els_rnid_ipvers_val[] = {
+ {0, "None"},
+ {1, "IPv4"},
+ {2, "IPv6"},
+ {0, NULL}
+};
+
+static const value_string fc_prli_fc4_val[] = {
+ {FC_TYPE_SCSI , "FCP"},
+ {FC_TYPE_IP , "IP/FC"},
+ {FC_TYPE_LLCSNAP , "LLC_SNAP"},
+ {FC_TYPE_ELS , "Ext Link Svc"},
+ {FC_TYPE_FCCT , "FC_CT"},
+ {FC_TYPE_SWILS , "SW_ILS"},
+ {FC_TYPE_AL , "AL"},
+ {FC_TYPE_SNMP , "SNMP"},
+ {FC_TYPE_CMNSVC , "Common to all FC-4 Types"},
+ {0, NULL},
+};
+
+static const value_string cbind_addr_mode_vals[] = {
+ {0, "Address Translation mode"},
+ {1, "Address Transparent Mode"},
+ {0, NULL},
+};
+
+static const value_string cbind_status_vals[] = {
+ {0, "Success"},
+ {16, "Failed - Unspecified Reason"},
+ {18, "Failed - Connection ID invalid"},
+ {0, NULL},
+};
+
+static const value_string unbind_status_vals[] = {
+ {0, "Success"},
+ {16, "Failed - Unspecified Reason"},
+ {17, "Failed - No such device"},
+ {18, "Failed - iFCP session already exists"},
+ {19, "Failed - Lack of resources"},
+ {20, "Failed - Incompatible address translation mode"},
+ {21, "Failed - Incorrect protocol version"},
+ {22, "Failed - Gateway not synchronized"},
+ {0, NULL},
+};
+
+typedef struct _fcels_conv_key {
+ guint32 conv_idx;
+} fcels_conv_key_t;
+
+typedef struct _fcels_conv_data {
+ guint32 opcode;
+} fcels_conv_data_t;
+
+static wmem_map_t *fcels_req_hash = NULL;
+
+static dissector_handle_t fcsp_handle;
+
+/*
+ * Hash Functions
+ */
+static gint
+fcels_equal(gconstpointer v, gconstpointer w)
+{
+ const fcels_conv_key_t *v1 = (const fcels_conv_key_t *)v;
+ const fcels_conv_key_t *v2 = (const fcels_conv_key_t *)w;
+
+ return (v1->conv_idx == v2->conv_idx);
+}
+
+static guint
+fcels_hash (gconstpointer v)
+{
+ const fcels_conv_key_t *key = (const fcels_conv_key_t *)v;
+ guint val;
+
+ val = key->conv_idx;
+
+ return val;
+}
+
+static const true_false_string tfs_fc_fcels_cmn_b2b = {
+ "Alt B2B Credit Mgmt",
+ "Normal B2B Credit Mgmt"
+};
+static const true_false_string tfs_fc_fcels_cmn_e_d_tov = {
+ "E_D_TOV Resolution in ns",
+ "E_D_TOV Resolution in ms"
+};
+static const true_false_string tfs_fc_fcels_cmn_seqcnt = {
+ "Cont. Incr SEQCNT rules",
+ "Normal SEQCNT rules"
+};
+static const true_false_string tfs_fc_fcels_cmn_payload = {
+ "Payload Len=256 bytes",
+ "Payload Len=116 bytes"
+};
+
+static void
+dissect_cmnsvc (proto_tree *parent_tree, tvbuff_t *tvb, int offset, guint16 flags, guint8 opcode)
+{
+ static int * const common_flags[] = {
+ &hf_fcels_cmn_cios,
+ &hf_fcels_cmn_rro,
+ &hf_fcels_cmn_vvv,
+ &hf_fcels_cmn_b2b,
+ &hf_fcels_cmn_multicast,
+ &hf_fcels_cmn_broadcast,
+ &hf_fcels_cmn_security,
+ &hf_fcels_cmn_clk,
+ &hf_fcels_cmn_dhd,
+ &hf_fcels_cmn_payload,
+ NULL
+ };
+
+ static int * const pflags[] = {
+ &hf_fcels_cmn_cios,
+ &hf_fcels_cmn_rro,
+ &hf_fcels_cmn_vvv,
+ &hf_fcels_cmn_b2b,
+ &hf_fcels_cmn_e_d_tov,
+ &hf_fcels_cmn_simplex,
+ &hf_fcels_cmn_multicast,
+ &hf_fcels_cmn_broadcast,
+ &hf_fcels_cmn_security,
+ &hf_fcels_cmn_clk,
+ &hf_fcels_cmn_dhd,
+ &hf_fcels_cmn_seqcnt,
+ &hf_fcels_cmn_payload,
+ NULL
+ };
+
+
+ if ((opcode == FC_ELS_PLOGI) || (opcode == FC_ELS_PDISC)) {
+ proto_tree_add_bitmask_value_with_flags(parent_tree, tvb, offset, hf_fcels_cmnfeatures,
+ ett_fcels_cmnfeatures, pflags, flags, BMT_NO_FLAGS);
+ } else {
+ proto_tree_add_bitmask_value_with_flags(parent_tree, tvb, offset, hf_fcels_cmnfeatures,
+ ett_fcels_cmnfeatures, common_flags, flags, BMT_NO_FLAGS);
+ }
+}
+
+
+static const true_false_string tfs_fc_fcels_cls_sdr = {
+ "Seq Delivery Requested",
+ "Out of Order Delivery Requested"
+};
+static const true_false_string tfs_fc_fcels_cls_nzctl = {
+ "Non-zero CS_CTL Tolerated",
+ "Non-zero CS_CTL Maybe Tolerated"
+};
+
+/* The next 3 routines decode only Class 2 & Class 3 relevant bits */
+static void
+dissect_clssvc_flags (proto_tree *parent_tree, tvbuff_t *tvb, int offset, guint16 flags, guint8 opcode)
+{
+ static int * const common_flags[] = {
+ &hf_fcels_cls_cns,
+ &hf_fcels_cls_prio,
+ NULL
+ };
+
+ static int * const pflags[] = {
+ &hf_fcels_cls_cns,
+ &hf_fcels_cls_sdr,
+ &hf_fcels_cls_prio,
+ &hf_fcels_cls_nzctl,
+ NULL
+ };
+
+ if ((opcode == FC_ELS_FLOGI) || (opcode == FC_ELS_FDISC)) {
+ proto_tree_add_bitmask_value_with_flags(parent_tree, tvb, offset, hf_fcels_clsflags,
+ ett_fcels_clsflags, pflags, flags, BMT_NO_FLAGS);
+ } else {
+ proto_tree_add_bitmask_value_with_flags(parent_tree, tvb, offset, hf_fcels_clsflags,
+ ett_fcels_clsflags, common_flags, flags, BMT_NO_FLAGS);
+ }
+}
+
+static const true_false_string tfs_fc_fcels_fcpflags_retry = {
+ "Retry Possible",
+ "Retry NOT possible"
+};
+static const true_false_string tfs_fc_fcels_fcpflags_ccomp = {
+ "Confirmed Comp",
+ "Comp NOT confirmed"
+};
+static const true_false_string tfs_fc_fcels_fcpflags_datao = {
+ "Data Overlay",
+ "NO data overlay"
+};
+static const true_false_string tfs_fc_fcels_fcpflags_initiator = {
+ "Initiator",
+ "NOT an initiator"
+};
+static const true_false_string tfs_fc_fcels_fcpflags_target = {
+ "Target",
+ "NOT a target"
+};
+static const true_false_string tfs_fc_fcels_fcpflags_rdxr = {
+ "Rd Xfer_Rdy Dis",
+ "NO rd xfer_rdy dis"
+};
+static const true_false_string tfs_fc_fcels_fcpflags_wrxr = {
+ "Wr Xfer_Rdy Dis",
+ "NO wr xfer_rdy dis"
+};
+
+static void
+dissect_fcp_flags (proto_tree *parent_tree, tvbuff_t *tvb, int offset, guint32 flags, guint8 isreq)
+{
+ static int * const req_flags[] = {
+ &hf_fcels_fcpflags_trireq,
+ &hf_fcels_fcpflags_retry,
+ &hf_fcels_fcpflags_ccomp,
+ &hf_fcels_fcpflags_datao,
+ &hf_fcels_fcpflags_initiator,
+ &hf_fcels_fcpflags_target,
+ &hf_fcels_fcpflags_rdxr,
+ &hf_fcels_fcpflags_wrxr,
+ NULL
+ };
+
+ static int * const rep_flags[] = {
+ &hf_fcels_fcpflags_trirep,
+ &hf_fcels_fcpflags_retry,
+ &hf_fcels_fcpflags_ccomp,
+ &hf_fcels_fcpflags_datao,
+ &hf_fcels_fcpflags_initiator,
+ &hf_fcels_fcpflags_target,
+ &hf_fcels_fcpflags_rdxr,
+ &hf_fcels_fcpflags_wrxr,
+ NULL
+ };
+
+ if (isreq) {
+ proto_tree_add_bitmask_value_with_flags(parent_tree, tvb, offset, hf_fcels_fcpflags,
+ ett_fcels_fcpflags, req_flags, flags, BMT_NO_FALSE);
+ } else {
+ proto_tree_add_bitmask_value_with_flags(parent_tree, tvb, offset, hf_fcels_fcpflags,
+ ett_fcels_fcpflags, rep_flags, flags, BMT_NO_FALSE);
+ }
+}
+
+
+static void
+dissect_speed_flags (proto_tree *parent_tree, tvbuff_t *tvb, int offset, guint32 flags, int port)
+{
+ proto_item *item;
+ static int * const speed_flags[] = {
+ &hf_fcels_speedflags_1gb,
+ &hf_fcels_speedflags_2gb,
+ &hf_fcels_speedflags_4gb,
+ &hf_fcels_speedflags_10gb,
+ NULL
+ };
+
+ item = proto_tree_add_bitmask_value_with_flags(parent_tree, tvb, offset, hf_fcels_speedflags,
+ ett_fcels_speedflags, speed_flags, flags, BMT_NO_FALSE|BMT_NO_TFS);
+ proto_item_set_text(item, "Port Speed Capabilities (Port %u): 0x%04x", port, flags);
+}
+
+static const true_false_string tfs_fc_fcels_tprloflags_gprlo = {
+ "Global PRLO",
+ "NO global prlo"
+};
+static const true_false_string tfs_fc_fcels_prliloflags_ipe = {
+ "Image Pair Estd",
+ "Image pair NOT estd"
+};
+static const true_false_string tfs_fc_fcels_prliloflags_eip = {
+ "Est Image Pair & Exchg Svc Param",
+ "Exchange Svc Param Only"
+};
+
+static void
+dissect_prlilo_flags (proto_tree *parent_tree, tvbuff_t *tvb, int offset, int flags, guint8 opcode)
+{
+ static int * const tprlo_flags[] = {
+ &hf_fcels_tprloflags_opav,
+ &hf_fcels_tprloflags_rpav,
+ &hf_fcels_tprloflags_npv,
+ &hf_fcels_tprloflags_gprlo,
+ NULL
+ };
+
+ static int * const prli_flags[] = {
+ &hf_fcels_prliloflags_opav,
+ &hf_fcels_tprloflags_rpav,
+ &hf_fcels_prliloflags_ipe,
+ NULL
+ };
+
+ static int * const not_prli_flags[] = {
+ &hf_fcels_prliloflags_opav,
+ &hf_fcels_tprloflags_rpav,
+ &hf_fcels_prliloflags_eip,
+ NULL
+ };
+
+ if (opcode == FC_ELS_TPRLO) {
+ proto_tree_add_bitmask_value_with_flags(parent_tree, tvb, offset, hf_fcels_prliloflags,
+ ett_fcels_prliloflags, tprlo_flags, flags, BMT_NO_FALSE|BMT_NO_TFS);
+
+ } else { /* opcode != TPRLO */
+ if (opcode == FC_ELS_PRLI) {
+ proto_tree_add_bitmask_value_with_flags(parent_tree, tvb, offset, hf_fcels_prliloflags,
+ ett_fcels_prliloflags, prli_flags, flags, BMT_NO_FALSE);
+ } else {
+ proto_tree_add_bitmask_value_with_flags(parent_tree, tvb, offset, hf_fcels_prliloflags,
+ ett_fcels_prliloflags, not_prli_flags, flags, BMT_NO_FALSE);
+ }
+ }
+}
+
+static const value_string initial_pa_vals[] = {
+ { 0, "Initial P_A Not Supported" },
+ { 1, "Initial P_A Supported" },
+ { 3, "Initial P_A Required & Supported" },
+ { 0, NULL }
+};
+static const true_false_string tfs_fc_fcels_initctl_ackgaa = {
+ "ACK Generation Assistance Avail",
+ "NO ack generation assistance"
+};
+
+static void
+dissect_initctl_flags (proto_tree *parent_tree, tvbuff_t *tvb, int offset, guint16 flags, guint8 opcode)
+{
+ static int * const plogi_flags[] = {
+ &hf_fcels_initctl_initial_pa,
+ &hf_fcels_initctl_ack0,
+ &hf_fcels_initctl_ackgaa,
+ &hf_fcels_initctl_sync,
+ NULL
+ };
+
+ static int * const not_plogi_flags[] = {
+ &hf_fcels_initctl_sync,
+ NULL
+ };
+
+ if ((opcode == FC_ELS_PLOGI) || (opcode == FC_ELS_PDISC)) {
+ proto_tree_add_bitmask_value_with_flags(parent_tree, tvb, offset, hf_fcels_initctl,
+ ett_fcels_initctl, plogi_flags, flags, BMT_NO_FALSE);
+ } else {
+ proto_tree_add_bitmask_value_with_flags(parent_tree, tvb, offset, hf_fcels_initctl,
+ ett_fcels_initctl, not_plogi_flags, flags, BMT_NO_FALSE);
+ }
+}
+
+
+static const value_string rcptctl_policy_vals[] = {
+ { 0, "Error Policy: Discard Policy only" },
+ { 1, "Error Policy: Reserved" },
+ { 2, "Error Policy: Both discard and process policies supported" },
+ { 3, "Error Policy: Reserved" },
+ { 0, NULL }
+};
+static const value_string rcptctl_category_vals[] = {
+ { 0, "1 Category/Seq" },
+ { 1, "2 Categories/Seq" },
+ { 3, "More than 2 Categories/Seq" },
+ { 0, NULL }
+};
+
+static void
+dissect_rcptctl_flags (proto_tree *parent_tree, tvbuff_t *tvb, int offset, guint16 flags, guint8 opcode)
+{
+ static int * const plogi_flags[] = {
+ &hf_fcels_rcptctl_ack0,
+ &hf_fcels_rcptctl_interlock,
+ &hf_fcels_rcptctl_policy,
+ &hf_fcels_rcptctl_category,
+ &hf_fcels_rcptctl_sync,
+ NULL
+ };
+
+ static int * const not_plogi_flags[] = {
+ &hf_fcels_rcptctl_sync,
+ NULL
+ };
+
+ if ((opcode == FC_ELS_PLOGI) || (opcode == FC_ELS_PDISC)) {
+ proto_tree_add_bitmask_value_with_flags(parent_tree, tvb, offset, hf_fcels_rcptctl,
+ ett_fcels_rcptctl, plogi_flags, flags, BMT_NO_FALSE);
+ } else {
+ proto_tree_add_bitmask_value_with_flags(parent_tree, tvb, offset, hf_fcels_rcptctl,
+ ett_fcels_rcptctl, not_plogi_flags, flags, BMT_NO_FALSE);
+ }
+}
+
+/* Maximum length of possible string from, construct_*_string
+ * 296 bytes, FIX possible buffer overflow */
+#define FCELS_LOGI_MAXSTRINGLEN 512
+
+static void
+dissect_fcels_logi (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ proto_item *ti, guint8 opcode)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ int offset = 0,
+ svcvld = 0,
+ svcclass;
+ proto_tree *logi_tree, *cmnsvc_tree;
+ guint16 flag;
+
+ if (tree) {
+ logi_tree = proto_item_add_subtree (ti, ett_fcels_logi);
+ proto_tree_add_item (logi_tree, hf_fcels_opcode, tvb, offset, 1, ENC_BIG_ENDIAN);
+
+ cmnsvc_tree = proto_tree_add_subtree(logi_tree, tvb, offset+4, 16, ett_fcels_logi_cmnsvc, NULL, "Common Svc Parameters");
+ proto_tree_add_item (cmnsvc_tree, hf_fcels_b2b, tvb, offset+6, 2, ENC_BIG_ENDIAN);
+ flag = tvb_get_ntohs (tvb, offset+8);
+
+ if (flag & 0x0001) {
+ svcvld = 1;
+ }
+
+ dissect_cmnsvc (cmnsvc_tree, tvb, offset+8, flag, opcode);
+
+ proto_tree_add_item (cmnsvc_tree, hf_fcels_bbscnum, tvb, offset+10, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (cmnsvc_tree, hf_fcels_rcvsize, tvb, offset+10, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item (cmnsvc_tree, hf_fcels_maxconseq, tvb, offset+12, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item (cmnsvc_tree, hf_fcels_reloffset, tvb, offset+14, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item (cmnsvc_tree, hf_fcels_edtov, tvb, offset+16, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item (cmnsvc_tree, hf_fcels_npname, tvb, offset+20, 8, ENC_NA);
+ proto_tree_add_item (cmnsvc_tree, hf_fcels_fnname, tvb, offset+28, 8, ENC_NA);
+
+ /* Add subtree for class paramters */
+ offset = 36;
+ for (svcclass = 1; svcclass < 5; svcclass++) {
+ cmnsvc_tree = proto_tree_add_subtree_format(logi_tree, tvb, offset, 16,
+ ett_fcels_logi_cmnsvc, NULL, "Class %d Svc Parameters", svcclass);
+
+ flag = tvb_get_ntohs (tvb, offset);
+ dissect_clssvc_flags (cmnsvc_tree, tvb, offset, flag, opcode);
+ if (flag & 0x8000) {
+ flag = tvb_get_ntohs (tvb, offset+2);
+ dissect_initctl_flags (cmnsvc_tree, tvb, offset+2, flag, opcode);
+
+ flag = tvb_get_ntohs (tvb, offset+4);
+ dissect_rcptctl_flags (cmnsvc_tree, tvb, offset+4, flag, opcode);
+
+ proto_tree_add_item (cmnsvc_tree, hf_fcels_clsrcvsize, tvb,
+ offset+6, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item (cmnsvc_tree, hf_fcels_conseq, tvb,
+ offset+8, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item (cmnsvc_tree, hf_fcels_e2e, tvb,
+ offset+10, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item (cmnsvc_tree, hf_fcels_openseq, tvb,
+ offset+12, 2, ENC_BIG_ENDIAN);
+ }
+ offset += 16;
+ }
+ proto_tree_add_item (logi_tree, hf_fcels_vendorvers, tvb, offset, 16, ENC_NA);
+ if (svcvld) {
+ proto_tree_add_item (logi_tree, hf_fcels_svcavail, tvb, offset+32, 8, ENC_NA);
+ }
+ }
+}
+
+static void
+dissect_fcels_plogi (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree,
+ guint8 isreq _U_, proto_item *ti)
+{
+ dissect_fcels_logi (tvb, pinfo, tree, ti, FC_ELS_PLOGI);
+}
+
+static void
+dissect_fcels_flogi (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree,
+ guint8 isreq _U_, proto_item *ti)
+{
+ dissect_fcels_logi (tvb, pinfo, tree, ti, FC_ELS_FLOGI);
+}
+
+static void
+dissect_fcels_logout (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ guint8 isreq, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ int offset = 5; /* bypass opcode+rsvd field */
+ proto_tree *logo_tree;
+
+ if (tree) {
+ logo_tree = proto_item_add_subtree (ti, ett_fcels_logo);
+
+ proto_tree_add_item (logo_tree, hf_fcels_opcode, tvb, offset-5, 1, ENC_BIG_ENDIAN);
+
+ if (!isreq) {
+ /* Accept has no payload */
+ return;
+ }
+
+ proto_tree_add_item (logo_tree, hf_fcels_nportid, tvb, offset, 3, ENC_NA);
+ proto_tree_add_item (logo_tree, hf_fcels_npname, tvb, offset+3, 8, ENC_NA);
+ }
+}
+
+static void
+dissect_fcels_abtx (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ guint8 isreq, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ int offset = 0;
+ proto_tree *abtx_tree;
+
+ if (tree) {
+ abtx_tree = proto_item_add_subtree (ti, ett_fcels_abtx);
+
+ proto_tree_add_item (abtx_tree, hf_fcels_opcode, tvb, offset, 1, ENC_BIG_ENDIAN);
+
+ if (!isreq) {
+ return;
+ }
+
+ proto_tree_add_item(abtx_tree, hf_fcels_recovery_qualifier_status, tvb, offset+4, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (abtx_tree, hf_fcels_nportid, tvb, offset+5, 3, ENC_NA);
+ proto_tree_add_item (abtx_tree, hf_fcels_oxid, tvb, offset+8, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item (abtx_tree, hf_fcels_rxid, tvb, offset+10, 2, ENC_BIG_ENDIAN);
+ }
+}
+
+static void
+dissect_fcels_rsi (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ guint8 isreq, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ int offset = 4;
+ proto_tree *rsi_tree;
+
+ if (tree) {
+ rsi_tree = proto_item_add_subtree (ti, ett_fcels_rsi);
+
+ proto_tree_add_item (rsi_tree, hf_fcels_opcode, tvb, offset-4, 1, ENC_BIG_ENDIAN);
+ if (!isreq)
+ return;
+
+ proto_tree_add_item (rsi_tree, hf_fcels_recovqual, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rsi_tree, hf_fcels_nportid, tvb, offset+1, 3, ENC_NA);
+ proto_tree_add_item (rsi_tree, hf_fcels_rxid, tvb, offset+4, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rsi_tree, hf_fcels_oxid, tvb, offset+6, 2, ENC_BIG_ENDIAN);
+ }
+}
+
+static void
+dissect_fcels_rrq (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ guint8 isreq, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ int offset = 0;
+ proto_tree *rrq_tree;
+
+ if (tree) {
+ rrq_tree = proto_item_add_subtree (ti, ett_fcels_rrq);
+
+ proto_tree_add_item (rrq_tree, hf_fcels_opcode, tvb, offset, 1, ENC_BIG_ENDIAN);
+ if (!isreq)
+ return;
+
+ proto_tree_add_item (rrq_tree, hf_fcels_nportid, tvb, offset+5, 3, ENC_NA);
+ proto_tree_add_item (rrq_tree, hf_fcels_oxid, tvb, offset+8, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rrq_tree, hf_fcels_rxid, tvb, offset+10, 2, ENC_BIG_ENDIAN);
+ }
+}
+
+static void
+dissect_fcels_rec (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ guint8 isreq, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ int offset = 0;
+ proto_tree *rec_tree;
+
+ if (tree) {
+ rec_tree = proto_item_add_subtree (ti, ett_fcels_rec);
+
+ proto_tree_add_item (rec_tree, hf_fcels_opcode, tvb, offset, 1, ENC_BIG_ENDIAN);
+
+ if (isreq) {
+ proto_tree_add_item (rec_tree, hf_fcels_nportid, tvb,
+ offset+5, 3, ENC_NA);
+ proto_tree_add_item (rec_tree, hf_fcels_oxid, tvb,
+ offset+8, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rec_tree, hf_fcels_rxid, tvb,
+ offset+10, 2, ENC_BIG_ENDIAN);
+ } else {
+ proto_tree_add_item (rec_tree, hf_fcels_oxid, tvb,
+ offset+4, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rec_tree, hf_fcels_rxid, tvb,
+ offset+6, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rec_tree, hf_fcels_nportid, tvb,
+ offset+9, 3, ENC_NA);
+ proto_tree_add_item (rec_tree, hf_fcels_resportid, tvb,
+ offset+13, 3, ENC_NA);
+ proto_tree_add_item (rec_tree, hf_fcels_rec_fc4, tvb,
+ offset+16, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_bitmask (rec_tree, tvb, offset+20, hf_fcels_estat,
+ ett_fcels_estat, hf_fcels_estat_fields,
+ ENC_BIG_ENDIAN);
+ }
+ }
+}
+
+static void
+dissect_fcels_pdisc (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree,
+ guint8 isreq _U_, proto_item *ti)
+{
+ dissect_fcels_logi (tvb, pinfo, tree, ti, FC_ELS_PDISC);
+}
+
+static void
+dissect_fcels_fdisc (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree,
+ guint8 isreq _U_, proto_item *ti)
+{
+ dissect_fcels_logi (tvb, pinfo, tree, ti, FC_ELS_FDISC);
+}
+
+static void
+dissect_fcels_adisc (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ guint8 isreq _U_, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ int offset = 5;
+ proto_tree *adisc_tree;
+
+ if (tree) {
+ adisc_tree = proto_item_add_subtree (ti, ett_fcels_adisc);
+
+ proto_tree_add_item (adisc_tree, hf_fcels_opcode, tvb, offset-5, 1, ENC_BIG_ENDIAN);
+
+ proto_tree_add_item (adisc_tree, hf_fcels_hardaddr, tvb, offset, 3, ENC_NA);
+ proto_tree_add_item (adisc_tree, hf_fcels_npname, tvb, offset+3, 8, ENC_NA);
+ proto_tree_add_item (adisc_tree, hf_fcels_fnname, tvb, offset+11, 8, ENC_NA);
+ proto_tree_add_item (adisc_tree, hf_fcels_nportid, tvb, offset+20, 3, ENC_NA);
+ }
+
+}
+
+static void
+dissect_fcels_farp (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ proto_item *ti)
+{
+ int offset = 4;
+ proto_tree *farp_tree;
+
+ if (tree) {
+ farp_tree = proto_item_add_subtree (ti, ett_fcels_farp);
+
+ proto_tree_add_item (farp_tree, hf_fcels_opcode, tvb, offset-4, 1, ENC_BIG_ENDIAN);
+
+ proto_tree_add_item (farp_tree, hf_fcels_farp_matchcodept,
+ tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (farp_tree, hf_fcels_nportid, tvb, offset+1,
+ 3, ENC_NA);
+ proto_tree_add_item (farp_tree, hf_fcels_farp_respaction, tvb,
+ offset+4, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (farp_tree, hf_fcels_resportid, tvb, offset+5,
+ 3, ENC_NA);
+ proto_tree_add_item (farp_tree, hf_fcels_npname, tvb, offset+8, 8, ENC_NA);
+ proto_tree_add_item (farp_tree, hf_fcels_fnname, tvb, offset+16, 8, ENC_NA);
+ proto_tree_add_item (farp_tree, hf_fcels_respname, tvb, offset+24,
+ 8, ENC_NA);
+ proto_tree_add_item (farp_tree, hf_fcels_respnname, tvb, offset+32,
+ 8, ENC_NA);
+ proto_tree_add_item (farp_tree, hf_fcels_reqipaddr, tvb, offset+40,
+ 16, ENC_NA);
+ proto_tree_add_item (farp_tree, hf_fcels_respipaddr, tvb, offset+56,
+ 16, ENC_NA);
+ }
+}
+
+static void
+dissect_fcels_farp_req (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree,
+ guint8 isreq _U_, proto_item *ti)
+{
+ dissect_fcels_farp (tvb, pinfo, tree, ti);
+}
+
+static void
+dissect_fcels_farp_rply (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree,
+ guint8 isreq _U_, proto_item *ti)
+{
+ dissect_fcels_farp (tvb, pinfo, tree, ti);
+}
+
+static void
+dissect_fcels_rps (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ guint8 isreq, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ int offset = 3;
+ guint8 flag;
+ proto_tree *rps_tree;
+
+ flag = tvb_get_guint8 (tvb, offset);
+
+ if (tree) {
+ rps_tree = proto_item_add_subtree (ti, ett_fcels_rps);
+
+ if (isreq) {
+ proto_tree_add_item (rps_tree, hf_fcels_rps_flag, tvb, offset, 1, ENC_BIG_ENDIAN);
+
+ proto_tree_add_item (rps_tree, hf_fcels_opcode, tvb, offset-3, 1, ENC_BIG_ENDIAN);
+
+ if (flag & 0x2) {
+ proto_tree_add_item (rps_tree, hf_fcels_npname, tvb, offset+1, 8, ENC_NA);
+ }
+ else if (flag & 0x1) {
+ proto_tree_add_item (rps_tree, hf_fcels_rps_portnum, tvb,
+ offset+5, 3, ENC_BIG_ENDIAN);
+ }
+ }
+ else {
+ proto_tree_add_item (rps_tree, hf_fcels_rps_flag, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rps_tree, hf_fcels_rps_portstatus, tvb,
+ offset+3, 2, ENC_BIG_ENDIAN);
+ /* Next 6 fields are from Link Error Status Block (LESB) */
+ proto_tree_add_item(rps_tree, hf_fcels_link_failure_count, tvb, offset+5, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(rps_tree, hf_fcels_loss_of_sync_count, tvb, offset+9, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(rps_tree, hf_fcels_loss_of_signal_count, tvb, offset+13, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(rps_tree, hf_fcels_primitive_seq_protocol_err, tvb, offset+17, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(rps_tree, hf_fcels_invalid_xmission_word, tvb, offset+21, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(rps_tree, hf_fcels_invalid_crc_count, tvb, offset+25, 4, ENC_BIG_ENDIAN);
+ if (flag & 0x01) {
+ /* Next 6 fields are from L_Port Extension field */
+ proto_tree_add_item(rps_tree, hf_fcels_l_port_status, tvb, offset+31, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item(rps_tree, hf_fcels_lip_al_ps, tvb, offset+36, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(rps_tree, hf_fcels_lip_f7_initiated_count, tvb, offset+37, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(rps_tree, hf_fcels_lip_f7_received_count, tvb, offset+41, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(rps_tree, hf_fcels_lip_f8_initiated_count, tvb, offset+45, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(rps_tree, hf_fcels_lip_f8_received_count, tvb, offset+49, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(rps_tree, hf_fcels_lip_reset_initiated_count, tvb, offset+53, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(rps_tree, hf_fcels_lip_reset_received_count, tvb, offset+57, 4, ENC_BIG_ENDIAN);
+ }
+ }
+ }
+}
+
+static void
+dissect_fcels_rpl (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ guint8 isreq, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ int offset = 0;
+ proto_tree *rpl_tree, *pb_tree;
+ int loop;
+
+ if (tree) {
+ rpl_tree = proto_item_add_subtree (ti, ett_fcels_rpl);
+
+ proto_tree_add_item (rpl_tree, hf_fcels_opcode, tvb, offset, 1, ENC_BIG_ENDIAN);
+
+ if (isreq) {
+ proto_tree_add_item(rpl_tree, hf_fcels_rpl_max_size, tvb, offset+6, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item(rpl_tree, hf_fcels_rpl_index, tvb, offset+9, 3, ENC_BIG_ENDIAN);
+ }
+ else {
+ /* Reply consists of a header and a number of port blocks */
+ proto_tree_add_item(rpl_tree, hf_fcels_rpl_payload_length, tvb, offset+2, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item(rpl_tree, hf_fcels_rpl_list_length, tvb, offset+5, 3, ENC_BIG_ENDIAN);
+ proto_tree_add_item(rpl_tree, hf_fcels_rpl_index_of_i_port_block, tvb, offset+9, 3, ENC_BIG_ENDIAN);
+ offset = 12;
+ /* The following loop is for dissecting the port blocks */
+ for (loop = tvb_get_ntoh24 (tvb, 5); loop > 0; loop--) {
+ pb_tree = proto_tree_add_subtree_format(rpl_tree, tvb, offset+12, 16,
+ ett_fcels_rplpb, NULL, "Port Block %u", loop);
+
+ proto_tree_add_item(pb_tree, hf_fcels_rpl_physical_port, tvb, offset, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(pb_tree, hf_fcels_rpl_port_identifier, tvb, offset+5, 3, ENC_NA);
+ proto_tree_add_item(pb_tree, hf_fcels_rpl_port_name, tvb, offset+8, 8, ENC_NA);
+ offset += 16;
+ }
+ }
+ }
+}
+
+static void
+dissect_fcels_fan (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ guint8 isreq _U_, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ int offset = 5;
+ proto_tree *fan_tree;
+
+ if (tree) {
+ fan_tree = proto_item_add_subtree (ti, ett_fcels_fan);
+
+ proto_tree_add_item (fan_tree, hf_fcels_opcode, tvb, offset-5, 1, ENC_BIG_ENDIAN);
+
+ proto_tree_add_item (fan_tree, hf_fcels_fabricaddr, tvb, offset, 3, ENC_NA);
+ proto_tree_add_item (fan_tree, hf_fcels_fabricpname, tvb, offset+3,
+ 8, ENC_NA);
+ proto_tree_add_item (fan_tree, hf_fcels_fnname, tvb, offset+11, 8, ENC_NA);
+ }
+}
+
+static void
+dissect_fcels_rscn (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ guint8 isreq, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ int offset = 1;
+ proto_tree *rscn_tree, *rectree;
+ int numrec, plen, i;
+
+ if (tree) {
+ rscn_tree = proto_item_add_subtree (ti, ett_fcels_rscn);
+
+ proto_tree_add_item (rscn_tree, hf_fcels_opcode, tvb, offset-1, 1, ENC_BIG_ENDIAN);
+ if (!isreq)
+ return;
+
+ proto_tree_add_item(rscn_tree, hf_fcels_rscn_page_len, tvb, offset, 1, ENC_BIG_ENDIAN);
+ plen = tvb_get_ntohs (tvb, offset+1);
+ proto_tree_add_item(rscn_tree, hf_fcels_rscn_payload_len, tvb, offset+1, 2, ENC_BIG_ENDIAN);
+ numrec = (plen - 4)/4;
+
+ offset = 4;
+ for (i = 0; i < numrec; i++) {
+ rectree = proto_tree_add_subtree_format(rscn_tree, tvb, offset, 4,
+ ett_fcels_rscn_rec, NULL, "Affected N_Port Page %u", i);
+
+ proto_tree_add_item (rectree, hf_fcels_rscn_evqual, tvb, offset,
+ 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rectree, hf_fcels_rscn_addrfmt, tvb, offset,
+ 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rectree, hf_fcels_rscn_domain, tvb, offset+1,
+ 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rectree, hf_fcels_rscn_area, tvb, offset+2,
+ 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rectree, hf_fcels_rscn_port, tvb, offset+3,
+ 1, ENC_BIG_ENDIAN);
+ offset += 4;
+ }
+ }
+}
+
+static void
+dissect_fcels_scr (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ guint8 isreq, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ int offset = 7;
+ proto_tree *scr_tree;
+
+ if (tree) {
+ scr_tree = proto_item_add_subtree (ti, ett_fcels_scr);
+ proto_tree_add_item (scr_tree, hf_fcels_opcode, tvb, offset-7, 1, ENC_BIG_ENDIAN);
+ if (isreq)
+ proto_tree_add_item (scr_tree, hf_fcels_scrregn, tvb, offset, 1, ENC_BIG_ENDIAN);
+ }
+}
+
+static void
+dissect_fcels_rnft (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ guint8 isreq, proto_item *ti)
+{
+ int offset = 0;
+ guint16 numrec, i;
+ proto_tree *rnft_tree, *fc4_tree;
+
+ if (tree) {
+ rnft_tree = proto_item_add_subtree (ti, ett_fcels_rnft);
+
+ proto_tree_add_item (rnft_tree, hf_fcels_opcode, tvb, offset, 1, ENC_BIG_ENDIAN);
+
+ if (isreq) {
+ proto_tree_add_item(rnft_tree, hf_fcels_rnft_max_size, tvb, offset+2, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item(rnft_tree, hf_fcels_rnft_index, tvb, offset+7, 1, ENC_BIG_ENDIAN);
+ }
+ else {
+ proto_tree_add_item(rnft_tree, hf_fcels_rnft_payload_len, tvb, offset+2, 2, ENC_BIG_ENDIAN);
+ numrec = tvb_get_guint8 (tvb, offset+5);
+ proto_tree_add_item(rnft_tree, hf_fcels_rnft_list_length, tvb, offset+5, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(rnft_tree, hf_fcels_rnft_index_of_first_rec_in_list, tvb, offset+7, 1, ENC_BIG_ENDIAN);
+ offset = 8;
+ for (i = 0; i < numrec; i++) {
+ fc4_tree = proto_tree_add_subtree_format(rnft_tree, tvb, offset, 4,
+ ett_fcels_rnft_fc4, NULL, "FC-4 Entry #%u", i);
+
+ proto_tree_add_item (fc4_tree, hf_fcels_rnft_fc4type, tvb,
+ offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(fc4_tree, hf_fcels_rnft_fc4_qualifier, tvb, offset+1, 3, ENC_BIG_ENDIAN);
+ offset += 4;
+ }
+ }
+ }
+}
+
+static void
+dissect_fcels_lsts (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ guint8 isreq, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ int offset = 5;
+ proto_tree *lsts_tree;
+
+ if (tree) {
+ lsts_tree = proto_item_add_subtree (ti, ett_fcels_lsts);
+
+ proto_tree_add_item (lsts_tree, hf_fcels_opcode, tvb, offset-5, 1, ENC_BIG_ENDIAN);
+ if (isreq) {
+ /* In case of LSTS, the reply has the meat */
+ return;
+ }
+ proto_tree_add_item (lsts_tree, hf_fcels_failedrcvr, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (lsts_tree, hf_fcels_flacompliance, tvb, offset+1,
+ 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (lsts_tree, hf_fcels_loopstate, tvb, offset+2, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (lsts_tree, hf_fcels_publicloop_bmap, tvb, offset+3,
+ 16, ENC_NA);
+ proto_tree_add_item (lsts_tree, hf_fcels_pvtloop_bmap, tvb, offset+19,
+ 16, ENC_NA);
+ proto_tree_add_item (lsts_tree, hf_fcels_alpa_map, tvb, offset+35,
+ 128, ENC_NA);
+ }
+}
+
+/* Maximum length of possible string from, dissect_fcels_prlilo_payload
+ * 119 bytes, FIX possible buffer overflow */
+#define FCELS_PRLILO_MAXSTRINGLEN 256
+
+static void
+dissect_fcels_prlilo_payload (tvbuff_t *tvb, packet_info *pinfo _U_,
+ guint8 isreq, proto_item *ti, guint8 opcode)
+{
+ int offset = 0;
+ guint8 type;
+ proto_tree *prli_tree, *svcpg_tree;
+ int num_svcpg, payload_len, i, flag;
+
+ /* We're assuming that we're invoked only if tree is not NULL i.e.
+ * we don't do the usual "if (tree)" check here, the caller must.
+ */
+ prli_tree = proto_item_add_subtree (ti, ett_fcels_prli);
+
+ proto_tree_add_item (prli_tree, hf_fcels_opcode, tvb, offset, 1, ENC_BIG_ENDIAN);
+
+ proto_tree_add_item(prli_tree, hf_fcels_prlilo_page_length, tvb, offset+1, 1, ENC_BIG_ENDIAN);
+ payload_len = tvb_get_ntohs (tvb, offset+2);
+ proto_tree_add_item(prli_tree, hf_fcels_prlilo_payload_length, tvb, offset+2, 2, ENC_BIG_ENDIAN);
+ num_svcpg = payload_len/16;
+
+ offset = 4;
+ for (i = 0; i < num_svcpg; i++) {
+ svcpg_tree = proto_tree_add_subtree_format(prli_tree, tvb, offset, 16,
+ ett_fcels_prli_svcpg, NULL, "Service Parameter Page %u", i);
+
+ type = tvb_get_guint8 (tvb, offset);
+ proto_tree_add_item(svcpg_tree, hf_fcels_prlilo_type, tvb, offset, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(svcpg_tree, hf_fcels_prlilo_type_code_extension, tvb, offset+1, 1, ENC_BIG_ENDIAN);
+
+ flag = tvb_get_guint8 (tvb, offset+2);
+ dissect_prlilo_flags (svcpg_tree, tvb, offset+2, flag, opcode);
+
+ if (!isreq && (opcode != FC_ELS_TPRLO)) {
+ /* This is valid only for ACC */
+ proto_tree_add_item(svcpg_tree, hf_fcels_prlilo_response_code, tvb, offset+2, 1, ENC_BIG_ENDIAN);
+ }
+ if (opcode != FC_ELS_TPRLO) {
+ proto_tree_add_item(svcpg_tree, hf_fcels_prlilo_originator_pa, tvb, offset+4, 4, ENC_BIG_ENDIAN);
+ }
+ else {
+ proto_tree_add_item(svcpg_tree, hf_fcels_prlilo_3rd_party_originator_pa, tvb, offset+4, 4, ENC_BIG_ENDIAN);
+ }
+ proto_tree_add_item(svcpg_tree, hf_fcels_prlilo_responder_pa, tvb, offset+8, 4, ENC_BIG_ENDIAN);
+
+ if (type == FC_TYPE_SCSI) {
+ flag = tvb_get_ntohs (tvb, offset+14);
+ dissect_fcp_flags (svcpg_tree, tvb, offset+12, flag, isreq);
+ }
+ else if ((opcode == FC_ELS_PRLI) && !isreq) {
+ proto_tree_add_item(svcpg_tree, hf_fcels_prlilo_service_parameter_response, tvb, offset+12, 4, ENC_BIG_ENDIAN);
+ }
+ else if (opcode == FC_ELS_TPRLO) {
+ proto_tree_add_item(svcpg_tree, hf_fcels_prlilo_3rd_party_n_port_id, tvb, offset+13, 3, ENC_NA);
+ }
+ }
+}
+
+static void
+dissect_fcels_prli (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree,
+ guint8 isreq, proto_item *ti)
+{
+ if (tree) {
+ dissect_fcels_prlilo_payload (tvb, pinfo, isreq, ti, FC_ELS_PRLI);
+ }
+}
+
+static void
+dissect_fcels_prlo (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree,
+ guint8 isreq, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ if (tree) {
+ dissect_fcels_prlilo_payload (tvb, pinfo, isreq, ti, FC_ELS_PRLO);
+ }
+}
+
+static void
+dissect_fcels_tprlo (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree,
+ guint8 isreq, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+
+ if (tree) {
+ dissect_fcels_prlilo_payload (tvb, pinfo, isreq, ti, FC_ELS_TPRLO);
+ }
+}
+
+static void
+dissect_fcels_lirr (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ guint8 isreq _U_, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ int offset = 4;
+ proto_tree *lirr_tree;
+ guint8 lirr_fmt;
+
+ if (tree) {
+ lirr_tree = proto_item_add_subtree (ti, ett_fcels_lirr);
+
+ proto_tree_add_item (lirr_tree, hf_fcels_opcode, tvb, offset-4, 1, ENC_BIG_ENDIAN);
+
+ proto_tree_add_item(lirr_tree, hf_fcels_lirr_regn_function, tvb, offset, 1, ENC_BIG_ENDIAN);
+ lirr_fmt = tvb_get_guint8 (tvb, offset+1);
+ if (!lirr_fmt) {
+ /* This scheme is resorted to because the value 0 has a string in
+ * the value_string that is not what we want displayed here.
+ */
+ proto_tree_add_uint_format_value(lirr_tree, hf_fcels_lirr_regn_format, tvb, offset, 1, 0, "Common Format");
+ }
+ else {
+ proto_tree_add_item(lirr_tree, hf_fcels_lirr_regn_format, tvb, offset, 1, ENC_BIG_ENDIAN);
+ }
+ }
+}
+
+static const true_false_string tfs_srl_flag = { "Scan only specified FL Port", "Scan all loops in domain" };
+
+static void
+dissect_fcels_srl (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ guint8 isreq, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ int offset = 4;
+ proto_tree *srl_tree;
+
+ if (tree) {
+ srl_tree = proto_item_add_subtree (ti, ett_fcels_srl);
+
+ proto_tree_add_item (srl_tree, hf_fcels_opcode, tvb, offset-4, 1, ENC_BIG_ENDIAN);
+ if (!isreq)
+ return;
+
+ proto_tree_add_item(srl_tree, hf_fcels_srl_flag, tvb, offset, 1, ENC_NA);
+ proto_tree_add_item(srl_tree, hf_fcels_srl_fl_port_addr, tvb, offset+1, 3, ENC_NA);
+ }
+}
+
+static void
+dissect_fcels_rpsc (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ guint8 isreq, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ int offset = 2;
+ int num_entries, i, cap;
+ proto_tree *rpsc_tree;
+
+ if (tree) {
+ rpsc_tree = proto_item_add_subtree (ti, ett_fcels_rpsc);
+
+ proto_tree_add_item (rpsc_tree, hf_fcels_opcode, tvb, offset-2, 1, ENC_BIG_ENDIAN);
+ if (isreq)
+ return;
+
+ num_entries = tvb_get_ntohs (tvb, offset);
+ proto_tree_add_item(rpsc_tree, hf_fcels_rpsc_number_of_entries, tvb, offset, 2, ENC_BIG_ENDIAN);
+ offset = 4;
+ for (i = 0; i < num_entries; i++, offset+=4) {
+ cap = tvb_get_ntohs (tvb, offset);
+ dissect_speed_flags (rpsc_tree, tvb, offset, cap, i);
+
+ proto_tree_add_item(rpsc_tree, hf_fcels_rpsc_port_oper_speed, tvb, offset+2, 2, ENC_BIG_ENDIAN);
+ }
+ }
+}
+
+
+static void
+dissect_fcels_cbind (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree,
+ proto_item *ti)
+{
+ int offset = 0;
+ proto_tree *cbind_tree=NULL;
+
+ if (tree) {
+ cbind_tree = proto_item_add_subtree (ti, ett_fcels_cbind);
+
+ proto_tree_add_item (cbind_tree, hf_fcels_opcode, tvb, offset, 1, ENC_BIG_ENDIAN);
+ }
+ col_set_str(pinfo->cinfo, COL_INFO, "CBIND ");
+
+ proto_tree_add_item (cbind_tree, hf_fcels_cbind_liveness, tvb, offset+4, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item (cbind_tree, hf_fcels_cbind_addr_mode, tvb, offset+6, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (cbind_tree, hf_fcels_cbind_ifcp_version, tvb, offset+7, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (cbind_tree, hf_fcels_cbind_userinfo, tvb, offset+8, 4, ENC_BIG_ENDIAN);
+
+ proto_tree_add_item (cbind_tree, hf_fcels_cbind_snpname, tvb, offset+12, 8, ENC_NA);
+ proto_tree_add_item (cbind_tree, hf_fcels_cbind_dnpname, tvb, offset+20, 8, ENC_NA);
+
+ switch(tvb_reported_length(tvb)){
+ case 32: /* 28 byte Request + 4 bytes FC CRC */
+ col_append_str (pinfo->cinfo, COL_INFO, "Request");
+ break;
+ case 40: /* 36 byte Response + 4 bytes FC CRC */
+ col_append_str (pinfo->cinfo, COL_INFO, "Response");
+ proto_tree_add_item (cbind_tree, hf_fcels_cbind_status, tvb, offset+30, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item (cbind_tree, hf_fcels_chandle, tvb, offset+34, 2, ENC_BIG_ENDIAN);
+ break;
+ }
+
+}
+
+static void
+dissect_fcels_unbind (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree,
+ proto_item *ti)
+{
+ int offset = 0;
+ proto_tree *cbind_tree=NULL;
+
+ if (tree) {
+ cbind_tree = proto_item_add_subtree (ti, ett_fcels_cbind);
+
+ proto_tree_add_item (cbind_tree, hf_fcels_opcode, tvb, offset, 1, ENC_BIG_ENDIAN);
+ }
+ col_set_str(pinfo->cinfo, COL_INFO, "UNBIND ");
+
+ proto_tree_add_item (cbind_tree, hf_fcels_cbind_userinfo, tvb, offset+4, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item (cbind_tree, hf_fcels_chandle, tvb, offset+10, 2, ENC_BIG_ENDIAN);
+
+
+ switch(tvb_reported_length(tvb)){
+ case 24: /* 20 byte Request + 4 bytes FC CRC */
+ col_append_str (pinfo->cinfo, COL_INFO, "Request");
+ break;
+ case 28: /* 24 byte Response + 4 bytes FC CRC */
+ col_append_str (pinfo->cinfo, COL_INFO, "Response");
+ proto_tree_add_item (cbind_tree, hf_fcels_unbind_status, tvb, offset+22, 2, ENC_BIG_ENDIAN);
+ break;
+ }
+
+}
+
+static void
+dissect_fcels_rnid (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ guint8 isreq, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ int offset = 0;
+ int clen;
+ proto_tree *rnid_tree;
+
+ if (tree) {
+ rnid_tree = proto_item_add_subtree (ti, ett_fcels_rnid);
+
+ proto_tree_add_item (rnid_tree, hf_fcels_opcode, tvb, offset, 1, ENC_BIG_ENDIAN);
+ if (isreq) {
+ proto_tree_add_item (rnid_tree, hf_fcels_nodeidfmt, tvb, offset+4,
+ 1, ENC_BIG_ENDIAN);
+ }
+ else {
+ /* We only decode responses to nodeid fmt DF */
+ proto_tree_add_item (rnid_tree, hf_fcels_nodeidfmt, tvb, offset+4,
+ 1, ENC_BIG_ENDIAN);
+ clen = tvb_get_guint8 (tvb, offset+5);
+ proto_tree_add_item(rnid_tree, hf_fcels_common_identification_data_length, tvb, offset+5, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rnid_tree, hf_fcels_spidlen, tvb, offset+7,
+ 1, ENC_BIG_ENDIAN);
+ if (clen) {
+ proto_tree_add_item (rnid_tree, hf_fcels_npname, tvb,
+ offset+8, 8, ENC_NA);
+ proto_tree_add_item (rnid_tree, hf_fcels_fnname, tvb,
+ offset+16, 8, ENC_NA);
+ }
+ if (tvb_get_guint8 (tvb, offset+4) == 0xDF) {
+ /* Decode the Specific Node ID Format as this is known */
+ proto_tree_add_item (rnid_tree, hf_fcels_vendoruniq, tvb,
+ offset+24, 16, ENC_NA);
+ proto_tree_add_item (rnid_tree, hf_fcels_asstype, tvb,
+ offset+40, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rnid_tree, hf_fcels_physport, tvb,
+ offset+44, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rnid_tree, hf_fcels_attnodes, tvb,
+ offset+48, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rnid_tree, hf_fcels_nodemgmt, tvb,
+ offset+52, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rnid_tree, hf_fcels_ipvers, tvb,
+ offset+53, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rnid_tree, hf_fcels_tcpport, tvb,
+ offset+54, 2, ENC_BIG_ENDIAN);
+ proto_tree_add_item (rnid_tree, hf_fcels_ip, tvb, offset+56,
+ 16, ENC_NA);
+ proto_tree_add_item (rnid_tree, hf_fcels_vendorsp, tvb,
+ offset+74, 2, ENC_BIG_ENDIAN);
+ }
+ }
+ }
+}
+
+static void
+dissect_fcels_rlir (tvbuff_t *tvb _U_, packet_info *pinfo _U_,
+ proto_tree *tree, guint8 isreq _U_,
+ proto_item *ti _U_)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+
+ if (tree) {
+ }
+}
+
+static void
+dissect_fcels_lsrjt (tvbuff_t *tvb, packet_info *pinfo _U_, proto_tree *tree,
+ guint8 isreq _U_, proto_item *ti)
+{
+ /* Set up structures needed to add the protocol subtree and manage it */
+ int offset = 5;
+ proto_tree *lsrjt_tree;
+
+ if (tree) {
+ lsrjt_tree = proto_item_add_subtree (ti, ett_fcels_lsrjt);
+
+ proto_tree_add_item (lsrjt_tree, hf_fcels_opcode, tvb, offset-5, 1, ENC_BIG_ENDIAN);
+
+ proto_tree_add_item (lsrjt_tree, hf_fcels_rjtcode, tvb, offset++, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (lsrjt_tree, hf_fcels_rjtdetcode, tvb, offset++, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item (lsrjt_tree, hf_fcels_vnduniq, tvb, offset, 1, ENC_BIG_ENDIAN);
+ }
+}
+
+static int
+dissect_fcels (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void* data)
+{
+
+/* Set up structures needed to add the protocol subtree and manage it */
+ proto_item *ti = NULL;
+ proto_tree *acc_tree;
+ guint8 isreq = FC_ELS_REQ;
+ int offset = 0;
+ guint8 opcode,
+ failed_opcode = 0;
+ conversation_t *conversation;
+ fcels_conv_data_t *cdata;
+ fcels_conv_key_t ckey, *req_key;
+ guint find_options, new_options;
+ address dstaddr;
+ guint8 addrdata[3];
+ fc_hdr *fchdr;
+
+ /* Reject the packet if data is NULL */
+ if (data == NULL)
+ return 0;
+ fchdr = (fc_hdr *)data;
+
+ /* Make entries in Protocol column and Info column on summary display */
+ col_set_str(pinfo->cinfo, COL_PROTOCOL, "FC ELS");
+ /* col_clear(pinfo->cinfo, COL_INFO); XXX: It seems to me that COL_INFO should be cleared here ?? */
+
+ /* decoding of this is done by each individual opcode handler */
+ opcode = tvb_get_guint8 (tvb, 0);
+
+ if (tree) {
+ ti = proto_tree_add_protocol_format (tree, proto_fcels, tvb, 0,
+ -1, "FC ELS");
+ }
+
+ /* Register conversation in case this is not a response */
+ if ((opcode != FC_ELS_LSRJT) && (opcode != FC_ELS_ACC)) {
+ if (opcode == FC_ELS_FLOGI) {
+ const guint8 *srcfc;
+
+ /* Check that the source address is, in fact, an FC address */
+ if (pinfo->src.type != AT_FC) {
+ expert_add_info_format(pinfo, ti, &ei_fcels_src_unknown,
+ "Unknown source address type: %u",
+ pinfo->src.type);
+ return 0;
+ }
+
+ srcfc = (const guint8 *)pinfo->src.data;
+ if (srcfc[2]) {
+ /* If it is a loop port, we'll need to remember the ALPA */
+ find_options = NO_PORT_B;
+ new_options = NO_PORT2;
+ }
+ else {
+ find_options = NO_PORT_B | NO_ADDR_B;
+ new_options = NO_PORT2 | NO_ADDR2;
+ }
+ }
+ else {
+ find_options = NO_PORT_B;
+ new_options = NO_PORT2;
+ }
+ conversation = find_conversation (pinfo->num, &pinfo->dst, &pinfo->src,
+ conversation_pt_to_conversation_type(pinfo->ptype), fchdr->oxid,
+ fchdr->rxid, find_options);
+
+ if (!conversation) {
+ conversation = conversation_new (pinfo->num, &pinfo->dst, &pinfo->src,
+ conversation_pt_to_conversation_type(pinfo->ptype), fchdr->oxid,
+ fchdr->rxid, new_options);
+ }
+
+ ckey.conv_idx = conversation->conv_index;
+
+ cdata = (fcels_conv_data_t *)wmem_map_lookup (fcels_req_hash,
+ &ckey);
+ 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->opcode = opcode;
+ }
+ else {
+ req_key = wmem_new(wmem_file_scope(), fcels_conv_key_t);
+ req_key->conv_idx = conversation->conv_index;
+
+ cdata = wmem_new(wmem_file_scope(), fcels_conv_data_t);
+ cdata->opcode = opcode;
+
+ wmem_map_insert (fcels_req_hash, req_key, cdata);
+ }
+ }
+ else {
+ isreq = FC_ELS_RPLY;
+
+ find_options = NO_PORT_B;
+ conversation = find_conversation (pinfo->num, &pinfo->dst, &pinfo->src,
+ conversation_pt_to_conversation_type(pinfo->ptype), fchdr->oxid,
+ fchdr->rxid, find_options);
+ if (!conversation) {
+ /* FLOGI has two ways to save state: without the src and using just
+ * the port (ALPA) part of the address. Try both.
+ */
+ const guint8 *dstfc;
+
+ /* Check that the source address is, in fact, an FC address */
+ if (pinfo->dst.type != AT_FC) {
+ expert_add_info_format(pinfo, ti, &ei_fcels_dst_unknown,
+ "Unknown destination address type: %u",
+ pinfo->dst.type);
+ return 0;
+ }
+
+ dstfc = (const guint8 *)pinfo->dst.data;
+
+ addrdata[0] = addrdata[1] = 0;
+ addrdata[2] = dstfc[2];
+ set_address (&dstaddr, AT_FC, 3, addrdata);
+ conversation = find_conversation (pinfo->num, &dstaddr, &pinfo->src,
+ conversation_pt_to_conversation_type(pinfo->ptype), fchdr->oxid,
+ fchdr->rxid, find_options);
+ }
+
+ if (!conversation) {
+ /* Finally check for FLOGI with both NO_PORT2 and NO_ADDR2 set */
+ find_options = NO_ADDR_B | NO_PORT_B;
+ conversation = find_conversation (pinfo->num, &pinfo->src, &pinfo->dst,
+ conversation_pt_to_conversation_type(pinfo->ptype), fchdr->oxid,
+ fchdr->rxid, find_options);
+ if (!conversation) {
+ if (tree && (opcode == FC_ELS_ACC)) {
+ /* No record of what this accept is for. Can't decode */
+ acc_tree = proto_item_add_subtree (ti, ett_fcels_acc);
+ proto_tree_add_expert(acc_tree, pinfo, &ei_fcels_no_record_of_exchange, tvb, offset, -1);
+ return 0;
+ }
+ failed_opcode = 0;
+ }
+ }
+
+ if (conversation) {
+ ckey.conv_idx = conversation->conv_index;
+
+ cdata = (fcels_conv_data_t *)wmem_map_lookup (fcels_req_hash, &ckey);
+
+ if (cdata != NULL) {
+ if ((find_options & NO_ADDR_B) && (cdata->opcode != FC_ELS_FLOGI)) {
+ /* only FLOGI can have this special check */
+ if (tree && (opcode == FC_ELS_ACC)) {
+ /* No record of what this accept is for. Can't decode */
+ acc_tree = proto_item_add_subtree (ti,
+ ett_fcels_acc);
+ proto_tree_add_expert(acc_tree, pinfo, &ei_fcels_no_record_of_exchange, tvb, offset, -1);
+ return 0;
+ }
+ }
+ if (opcode == FC_ELS_ACC)
+ opcode = cdata->opcode;
+ else
+ failed_opcode = cdata->opcode;
+ }
+
+ if (tree) {
+ if ((cdata == NULL) && (opcode != FC_ELS_LSRJT)) {
+ /* No record of what this accept is for. Can't decode */
+ acc_tree = proto_item_add_subtree (ti, ett_fcels_acc);
+ proto_tree_add_expert(acc_tree, pinfo, &ei_fcels_no_record_of_els_req, tvb, offset, -1);
+ return 0;
+ }
+ }
+ }
+ }
+
+ if (isreq == FC_ELS_REQ) {
+ col_add_str (pinfo->cinfo, COL_INFO,
+ val_to_str_ext (opcode, &fc_els_proto_val_ext, "0x%x"));
+ }
+ else if (opcode == FC_ELS_LSRJT) {
+ col_add_fstr (pinfo->cinfo, COL_INFO, "LS_RJT (%s)",
+ val_to_str_ext (failed_opcode, &fc_els_proto_val_ext, "0x%x"));
+ }
+ else {
+ col_add_fstr (pinfo->cinfo, COL_INFO, "ACC (%s)",
+ val_to_str_ext (opcode, &fc_els_proto_val_ext, "0x%x"));
+ }
+
+ switch (opcode) {
+ case FC_ELS_LSRJT:
+ dissect_fcels_lsrjt (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_PLOGI:
+ dissect_fcels_plogi (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_FLOGI:
+ dissect_fcels_flogi (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_LOGOUT:
+ dissect_fcels_logout (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_ABTX:
+ dissect_fcels_abtx (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_RSI:
+ dissect_fcels_rsi (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_RRQ:
+ dissect_fcels_rrq (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_REC:
+ dissect_fcels_rec (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_PRLI:
+ dissect_fcels_prli (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_PRLO:
+ dissect_fcels_prlo (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_TPRLO:
+ dissect_fcels_tprlo (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_PDISC:
+ dissect_fcels_pdisc (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_FDISC:
+ dissect_fcels_fdisc (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_ADISC:
+ dissect_fcels_adisc (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_FARP_REQ:
+ dissect_fcels_farp_req (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_FARP_RPLY:
+ dissect_fcels_farp_rply (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_RPS:
+ dissect_fcels_rps (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_RPL:
+ dissect_fcels_rpl (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_FAN:
+ dissect_fcels_fan (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_RSCN:
+ dissect_fcels_rscn (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_SCR:
+ dissect_fcels_scr (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_RNFT:
+ dissect_fcels_rnft (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_LSTS:
+ dissect_fcels_lsts (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_RNID:
+ dissect_fcels_rnid (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_RLIR:
+ dissect_fcels_rlir (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_LIRR:
+ dissect_fcels_lirr (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_SRL:
+ dissect_fcels_srl (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_RPSC:
+ dissect_fcels_rpsc (tvb, pinfo, tree, isreq, ti);
+ break;
+ case FC_ELS_AUTH:
+ if (isreq && fcsp_handle)
+ call_dissector (fcsp_handle, tvb, pinfo, tree);
+ break;
+ case FC_ELS_CBIND:
+ dissect_fcels_cbind (tvb, pinfo, tree, ti);
+ break;
+ case FC_ELS_UNBIND:
+ dissect_fcels_unbind (tvb, pinfo, tree, ti);
+ break;
+ default:
+ call_data_dissector(tvb, pinfo, tree);
+ break;
+ }
+
+ return tvb_reported_length(tvb);
+}
+
+void
+proto_register_fcels (void)
+{
+ static hf_register_info hf[] = {
+ { &hf_fcels_opcode,
+ {"Cmd Code", "fcels.opcode", FT_UINT8, BASE_HEX | BASE_EXT_STRING,
+ &fc_els_proto_val_ext, 0x0, NULL, HFILL}},
+ { &hf_fcels_rjtcode,
+ {"Reason Code", "fcels.rjt.reason", FT_UINT8, BASE_HEX | BASE_EXT_STRING,
+ &fc_els_rjt_val_ext, 0x0, NULL, HFILL}},
+ { &hf_fcels_rjtdetcode,
+ {"Reason Explanation", "fcels.rjt.detail", FT_UINT8, BASE_HEX | BASE_EXT_STRING,
+ &fc_els_rjt_det_val_ext, 0x0, NULL, HFILL}},
+ { &hf_fcels_vnduniq,
+ {"Vendor Unique", "fcels.rjt.vnduniq", FT_UINT8, BASE_HEX, NULL,
+ 0x0, NULL, HFILL}},
+ { &hf_fcels_b2b,
+ {"B2B Credit", "fcels.logi.b2b", FT_UINT16, BASE_DEC, NULL, 0x0, NULL,
+ HFILL}},
+ { &hf_fcels_cmnfeatures,
+ {"Common Svc Parameters", "fcels.logi.cmnfeatures", FT_UINT16, BASE_HEX, NULL,
+ 0x0, NULL, HFILL}},
+ { &hf_fcels_bbscnum,
+ {"BB_SC Number", "fcels.logi.bbscnum", FT_UINT8, BASE_DEC, NULL, 0xF0, NULL,
+ HFILL}},
+ { &hf_fcels_rcvsize,
+ {"Receive Size", "fcels.logi.rcvsize", FT_UINT16, BASE_DEC, NULL, 0x0FFF, NULL,
+ HFILL}},
+ { &hf_fcels_maxconseq,
+ {"Max Concurrent Seq", "fcels.logi.maxconseq", FT_UINT16, BASE_DEC, NULL,
+ 0x0, NULL, HFILL}},
+ { &hf_fcels_reloffset,
+ {"Relative Offset By Info Cat", "fcels.logi.reloff", FT_UINT16, BASE_DEC,
+ NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_edtov,
+ {"E_D_TOV", "fcels.edtov", FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_npname,
+ {"N_Port Port_Name", "fcels.npname", FT_FCWWN, BASE_NONE, NULL, 0x0,
+ NULL, HFILL}},
+ { &hf_fcels_fnname,
+ {"Fabric/Node Name", "fcels.fnname", FT_FCWWN, BASE_NONE, NULL, 0x0,
+ NULL, HFILL}},
+#if 0
+ { &hf_fcels_cls1param,
+ {"Class 1 Svc Param", "fcels.logi.cls1param", FT_BYTES, BASE_NONE, NULL, 0x0,
+ NULL, HFILL}},
+ { &hf_fcels_cls2param,
+ {"Class 2 Svc Param", "fcels.logi.cls2param", FT_BYTES, BASE_NONE, NULL, 0x0,
+ NULL, HFILL}},
+ { &hf_fcels_cls3param,
+ {"Class 3 Svc Param", "fcels.logi.cls3param", FT_BYTES, BASE_NONE, NULL, 0x0,
+ NULL, HFILL}},
+ { &hf_fcels_cls4param,
+ {"Class 4 Svc Param", "fcels.logi.cls4param", FT_BYTES, BASE_NONE, NULL, 0x0,
+ NULL, HFILL}},
+#endif
+ { &hf_fcels_vendorvers,
+ {"Vendor Version", "fcels.logi.vendvers", FT_BYTES, BASE_NONE, NULL, 0x0, NULL,
+ HFILL}},
+ { &hf_fcels_svcavail,
+ {"Services Availability", "fcels.logi.svcavail", FT_BYTES, BASE_NONE, NULL,
+ 0x0, NULL, HFILL}},
+ { &hf_fcels_clsflags,
+ {"Service Options", "fcels.logi.clsflags", FT_UINT16, BASE_HEX, NULL, 0x0, NULL,
+ HFILL}},
+ { &hf_fcels_clsrcvsize,
+ {"Class Recv Size", "fcels.logi.clsrcvsize", FT_UINT16, BASE_DEC, NULL,
+ 0x0, NULL, HFILL}},
+ { &hf_fcels_conseq,
+ {"Total Concurrent Seq", "fcels.logi.totconseq", FT_UINT16, BASE_DEC, NULL,
+ 0x0, NULL, HFILL}},
+ { &hf_fcels_e2e,
+ {"End2End Credit", "fcels.logi.e2e", FT_UINT16, BASE_DEC, NULL, 0x0, NULL,
+ HFILL}},
+ { &hf_fcels_openseq,
+ {"Open Seq Per Exchg", "fcels.logi.openseq", FT_UINT16, BASE_DEC, NULL, 0x0,
+ NULL, HFILL}},
+ { &hf_fcels_nportid,
+ {"Originator S_ID", "fcels.portid", FT_BYTES, SEP_DOT, NULL, 0x0,
+ NULL, HFILL}},
+ { &hf_fcels_oxid,
+ {"OXID", "fcels.oxid", FT_UINT16, BASE_HEX, NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_rxid,
+ {"RXID", "fcels.rxid", FT_UINT16, BASE_HEX, NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_recovqual,
+ {"Recovery Qualifier", "fcels.rcovqual", FT_UINT8, BASE_HEX, NULL,
+ 0x0, NULL, HFILL}},
+ { &hf_fcels_fabricaddr,
+ {"Fabric Address", "fcels.faddr", FT_BYTES, SEP_DOT, NULL, 0x0, NULL,
+ HFILL}},
+ { &hf_fcels_fabricpname,
+ {"Fabric Port Name", "fcels.fpname", FT_FCWWN, BASE_NONE, NULL, 0x0,
+ NULL, HFILL}},
+ { &hf_fcels_failedrcvr,
+ {"Failed Receiver AL_PA", "fcels.faildrcvr", FT_UINT8, BASE_HEX, NULL,
+ 0x0, NULL, HFILL}},
+ { &hf_fcels_flacompliance,
+ {"FC-FLA Compliance", "fcels.flacompliance", FT_UINT8, BASE_HEX,
+ VALS (fc_els_flacompliance_val), 0x0, NULL, HFILL}},
+ { &hf_fcels_loopstate,
+ {"Loop State", "fcels.loopstate", FT_UINT8, BASE_HEX,
+ VALS (fc_els_loopstate_val), 0x0, NULL, HFILL}},
+ { &hf_fcels_publicloop_bmap,
+ {"Public Loop Device Bitmap", "fcels.pubdev_bmap", FT_BYTES, BASE_NONE,
+ NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_pvtloop_bmap,
+ {"Private Loop Device Bitmap", "fcels.pvtdev_bmap", FT_BYTES,
+ BASE_NONE, NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_alpa_map,
+ {"AL_PA Map", "fcels.alpa", FT_BYTES, BASE_NONE, NULL, 0x0, NULL,
+ HFILL}},
+ { &hf_fcels_scrregn,
+ {"Registration Function", "fcels.scr.regn", FT_UINT8, BASE_HEX,
+ VALS (fc_els_scr_reg_val), 0x0, NULL, HFILL}},
+ { &hf_fcels_farp_matchcodept,
+ {"Match Address Code Points", "fcels.matchcp", FT_UINT8, BASE_DEC,
+ NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_farp_respaction,
+ {"Responder Action", "fcels.respaction", FT_UINT8, BASE_HEX,
+ VALS (fc_els_farp_respaction_val), 0x0, NULL, HFILL}},
+ { &hf_fcels_resportid,
+ {"Responding Port ID", "fcels.resportid", FT_BYTES, SEP_DOT,
+ NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_respname,
+ {"Responding Port Name", "fcels.respname", FT_FCWWN, BASE_NONE,
+ NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_respnname,
+ {"Responding Node Name", "fcels.respnname", FT_FCWWN, BASE_NONE,
+ NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_reqipaddr,
+ {"Requesting IP Address", "fcels.reqipaddr", FT_IPv6, BASE_NONE,
+ NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_respipaddr,
+ {"Responding IP Address", "fcels.respipaddr", FT_IPv6, BASE_NONE,
+ NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_hardaddr,
+ {"Hard Address of Originator", "fcels.hrdaddr", FT_BYTES, SEP_DOT,
+ NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_rps_flag,
+ {"Flag", "fcels.flag", FT_UINT8, BASE_HEX, NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_rps_portnum,
+ {"Physical Port Number", "fcels.portnum", FT_UINT32, BASE_HEX, NULL,
+ 0x0, NULL, HFILL}},
+ { &hf_fcels_rps_portstatus,
+ {"Port Status", "fcels.portstatus", FT_UINT16, BASE_HEX | BASE_EXT_STRING,
+ &fc_els_portstatus_val_ext, 0x0, NULL, HFILL}},
+ { &hf_fcels_rnft_fc4type,
+ {"FC-4 Type", "fcels.rnft.fc4type", FT_UINT8, BASE_HEX,
+ VALS (fc_fc4_val), 0x0, NULL, HFILL}},
+ { &hf_fcels_rscn_evqual,
+ {"Event Qualifier", "fcels.rscn.evqual", FT_UINT8, BASE_HEX,
+ VALS (fc_els_rscn_evqual_val), 0x3C, NULL, HFILL}},
+ { &hf_fcels_rscn_addrfmt,
+ {"Address Format", "fcels.rscn.addrfmt", FT_UINT8, BASE_HEX,
+ VALS (fc_els_rscn_addrfmt_val), 0x03, NULL, HFILL}},
+ { &hf_fcels_rscn_domain,
+ {"Affected Domain", "fcels.rscn.domain", FT_UINT8, BASE_HEX,
+ NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_rscn_area,
+ {"Affected Area", "fcels.rscn.area", FT_UINT8, BASE_HEX,
+ NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_rscn_port,
+ {"Affected Port", "fcels.rscn.port", FT_UINT8, BASE_HEX,
+ NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_rec_fc4,
+ {"FC4 value", "fcels.rec.fc4value", FT_UINT32, BASE_HEX,
+ NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_estat,
+ {"Exchange Status", "fcels.estat", FT_UINT32, BASE_HEX,
+ NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_estat_resp,
+ {"Sequence Responder", "fcels.estat.resp", FT_BOOLEAN, 32,
+ TFS(&tfs_fcels_estat_resp),
+ FC_ESB_ST_RESP, "Seq responder?", HFILL}},
+ { &hf_fcels_estat_seq_init,
+ {"Sequence Initiative", "fcels.estat.seq_init", FT_BOOLEAN, 32,
+ TFS(&tfs_fcels_estat_seq_init),
+ FC_ESB_ST_SEQ_INIT, "Responder has Sequence Initiative?", HFILL}},
+ { &hf_fcels_estat_compl,
+ {"Exchange Complete", "fcels.estat.complete", FT_BOOLEAN, 32,
+ TFS(&tfs_complete_incomplete),
+ FC_ESB_ST_COMPLETE, "Exchange complete?", HFILL}},
+ { &hf_fcels_nodeidfmt,
+ {"Node Identification Format", "fcels.rnid.nodeidfmt", FT_UINT8,
+ BASE_HEX, VALS (fc_els_nodeid_val), 0x0, NULL, HFILL}},
+ { &hf_fcels_spidlen,
+ {"Specific Id Length", "fcels.rnid.spidlen", FT_UINT8, BASE_DEC, NULL,
+ 0x0, NULL, HFILL}},
+ { &hf_fcels_vendoruniq,
+ {"Vendor Unique", "fcels.rnid.vendoruniq", FT_BYTES, BASE_NONE, NULL,
+ 0x0, NULL, HFILL}},
+ { &hf_fcels_vendorsp,
+ {"Vendor Specific", "fcels.rnid.vendorsp", FT_UINT16, BASE_HEX, NULL,
+ 0x0, NULL, HFILL}},
+ { &hf_fcels_asstype,
+ {"Associated Type", "fcels.rnid.asstype", FT_UINT32, BASE_HEX | BASE_EXT_STRING,
+ &fc_els_rnid_asstype_val_ext, 0x0, NULL, HFILL}},
+ { &hf_fcels_physport,
+ {"Physical Port Number", "fcels.rnid.physport", FT_UINT32, BASE_HEX,
+ NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_attnodes,
+ {"Number of Attached Nodes", "fcels.rnid.attnodes", FT_UINT32,
+ BASE_HEX, NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_nodemgmt,
+ {"Node Management", "fcels.rnid.nodemgmt", FT_UINT8, BASE_HEX,
+ VALS (fc_els_rnid_mgmt_val), 0x0, NULL, HFILL}},
+ { &hf_fcels_ipvers,
+ {"IP Version", "fcels.rnid.ipvers", FT_UINT8, BASE_HEX,
+ VALS (fc_els_rnid_ipvers_val), 0x0, NULL, HFILL}},
+ { &hf_fcels_tcpport,
+ {"TCP/UDP Port Number", "fcels.rnid.tcpport", FT_UINT16, BASE_DEC,
+ NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_ip,
+ {"IP Address", "fcels.rnid.ip", FT_IPv6, BASE_NONE, NULL, 0x0, NULL,
+ HFILL}},
+ { &hf_fcels_cbind_liveness,
+ {"Liveness Test Interval", "fcels.cbind.liveness", FT_UINT16, BASE_DEC,
+ NULL, 0x0, "Liveness Test Interval in seconds", HFILL}},
+ { &hf_fcels_cbind_addr_mode,
+ {"Addressing Mode", "fcels.cbind.addr_mode", FT_UINT8, BASE_HEX,
+ VALS (cbind_addr_mode_vals), 0x0, NULL, HFILL}},
+ { &hf_fcels_cbind_ifcp_version,
+ {"iFCP version", "fcels.cbind.ifcp_version", FT_UINT8, BASE_DEC,
+ NULL, 0x0, "Version of iFCP protocol", HFILL}},
+ { &hf_fcels_cbind_userinfo,
+ {"UserInfo", "fcels.cbind.userinfo", FT_UINT32, BASE_HEX,
+ NULL, 0x0, "Userinfo token", HFILL}},
+ { &hf_fcels_cbind_snpname,
+ {"Source N_Port Port_Name", "fcels.cbind.snpname", FT_FCWWN, BASE_NONE, NULL, 0x0,
+ NULL, HFILL}},
+ { &hf_fcels_cbind_dnpname,
+ {"Destination N_Port Port_Name", "fcels.cbind.dnpname", FT_FCWWN, BASE_NONE, NULL, 0x0,
+ NULL, HFILL}},
+ { &hf_fcels_cbind_status,
+ {"Status", "fcels.cbind.status", FT_UINT16, BASE_DEC,
+ VALS (cbind_status_vals), 0x0, "Cbind status", HFILL}},
+ { &hf_fcels_chandle,
+ {"Connection Handle", "fcels.cbind.handle", FT_UINT16, BASE_HEX,
+ NULL, 0x0, "Cbind/Unbind connection handle", HFILL}},
+ { &hf_fcels_unbind_status,
+ {"Status", "fcels.unbind.status", FT_UINT16, BASE_DEC,
+ VALS (unbind_status_vals), 0x0, "Unbind status", HFILL}},
+ { &hf_fcels_cmn_cios,
+ {"Cont. Incr. Offset Supported", "fcels.cmn.cios", FT_BOOLEAN, 16,
+ TFS(&tfs_supported_not_supported), 0x8000, NULL, HFILL}},
+ { &hf_fcels_cmn_rro,
+ {"RRO Supported", "fcels.cmn.rro", FT_BOOLEAN, 16,
+ TFS(&tfs_supported_not_supported), 0x4000, NULL, HFILL}},
+ { &hf_fcels_cmn_vvv,
+ {"Valid Vendor Version", "fcels.cmn.vvv", FT_BOOLEAN, 16,
+ TFS(&tfs_valid_invalid), 0x2000, NULL, HFILL}},
+ { &hf_fcels_cmn_b2b,
+ {"B2B Credit Mgmt", "fcels.cmn.bbb", FT_BOOLEAN, 16,
+ TFS(&tfs_fc_fcels_cmn_b2b), 0x0800, NULL, HFILL}},
+ { &hf_fcels_cmn_e_d_tov,
+ {"E_D_TOV", "fcels.cmn.e_d_tov", FT_BOOLEAN, 16,
+ TFS(&tfs_fc_fcels_cmn_e_d_tov), 0x0400, NULL, HFILL}},
+ { &hf_fcels_cmn_simplex,
+ {"Simplex", "fcels.cmn.simplex", FT_BOOLEAN, 16,
+ TFS(&tfs_supported_not_supported), 0x0040, NULL, HFILL}},
+ { &hf_fcels_cmn_multicast,
+ {"Multicast", "fcels.cmn.multicast", FT_BOOLEAN, 16,
+ TFS(&tfs_supported_not_supported), 0x0200, NULL, HFILL}},
+ { &hf_fcels_cmn_broadcast,
+ {"Broadcast", "fcels.cmn.broadcast", FT_BOOLEAN, 16,
+ TFS(&tfs_supported_not_supported), 0x0100, NULL, HFILL}},
+ { &hf_fcels_cmn_security,
+ {"Security", "fcels.cmn.security", FT_BOOLEAN, 16,
+ TFS(&tfs_set_notset), 0x0020, NULL, HFILL}},
+ { &hf_fcels_cmn_clk,
+ {"Clk Sync", "fcels.cmn.clk", FT_BOOLEAN, 16,
+ TFS(&tfs_capable_not_capable), 0x0010, NULL, HFILL}},
+ { &hf_fcels_cmn_dhd,
+ {"DHD Capable", "fcels.cmn.dhd", FT_BOOLEAN, 16,
+ TFS(&tfs_capable_not_capable), 0x0004, NULL, HFILL}},
+ { &hf_fcels_cmn_seqcnt,
+ {"SEQCNT", "fcels.cmn.seqcnt", FT_BOOLEAN, 16,
+ TFS(&tfs_fc_fcels_cmn_seqcnt), 0x0002, NULL, HFILL}},
+ { &hf_fcels_cmn_payload,
+ {"Payload Len", "fcels.cmn.payload", FT_BOOLEAN, 16,
+ TFS(&tfs_fc_fcels_cmn_payload), 0x0001, NULL, HFILL}},
+ { &hf_fcels_cls_cns,
+ {"Class Supported", "fcels.cls.cns", FT_BOOLEAN, 16,
+ TFS(&tfs_supported_not_supported), 0x8000, NULL, HFILL}},
+ { &hf_fcels_cls_sdr,
+ {"Delivery Mode", "fcels.cls.sdr", FT_BOOLEAN, 16,
+ TFS(&tfs_fc_fcels_cls_sdr), 0x0800, NULL, HFILL}},
+ { &hf_fcels_cls_prio,
+ {"Priority", "fcels.cls.prio", FT_BOOLEAN, 16,
+ TFS(&tfs_supported_not_supported), 0x0080, NULL, HFILL}},
+ { &hf_fcels_cls_nzctl,
+ {"Non-zero CS_CTL", "fcels.cls.nzctl", FT_BOOLEAN, 16,
+ TFS(&tfs_fc_fcels_cls_nzctl), 0x0040, NULL, HFILL}},
+ { &hf_fcels_initctl,
+ {"Initiator Ctl", "fcels.logi.initctl", FT_UINT16, BASE_HEX,
+ NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_initctl_initial_pa,
+ {"Initial P_A", "fcels.logi.initctl.initial_pa", FT_UINT16, BASE_HEX,
+ VALS (initial_pa_vals), 0x3000, NULL, HFILL}},
+ { &hf_fcels_initctl_ack0,
+ {"ACK0 Capable", "fcels.logi.initctl.ack0", FT_BOOLEAN, 16,
+ TFS(&tfs_capable_not_capable), 0x0800, NULL, HFILL}},
+ { &hf_fcels_initctl_ackgaa,
+ {"ACK GAA", "fcels.logi.initctl.ackgaa", FT_BOOLEAN, 16,
+ TFS(&tfs_fc_fcels_initctl_ackgaa), 0x0200, NULL, HFILL}},
+ { &hf_fcels_initctl_sync,
+ {"Clock Sync", "fcels.logi.initctl.sync", FT_BOOLEAN, 16,
+ TFS(&tfs_supported_not_supported), 0x0010, NULL, HFILL}},
+ { &hf_fcels_rcptctl,
+ {"Recipient Ctl", "fcels.logi.rcptctl", FT_UINT16, BASE_HEX,
+ NULL, 0x0, NULL, HFILL}},
+ { &hf_fcels_rcptctl_ack0,
+ {"ACK0", "fcels.logi.rcptctl.ack", FT_BOOLEAN, 16,
+ TFS(&tfs_supported_not_supported), 0x8000, NULL, HFILL}},
+ { &hf_fcels_rcptctl_interlock,
+ {"X_ID Interlock", "fcels.logi.rcptctl.interlock", FT_BOOLEAN, 16,
+ TFS(&tfs_requested_not_requested), 0x2000, NULL, HFILL}},
+ { &hf_fcels_rcptctl_policy,
+ {"Policy", "fcels.logi.rcptctl.policy", FT_UINT16, BASE_HEX,
+ VALS (rcptctl_policy_vals), 0x1800, NULL, HFILL}},
+ { &hf_fcels_rcptctl_category,
+ {"Category", "fcels.logi.rcptctl.category", FT_UINT16, BASE_HEX,
+ VALS (rcptctl_category_vals), 0x0030, NULL, HFILL}},
+ { &hf_fcels_rcptctl_sync,
+ {"Clock Sync", "fcels.logi.rcptctl.sync", FT_BOOLEAN, 16,
+ TFS(&tfs_supported_not_supported), 0x0008, NULL, HFILL}},
+ { &hf_fcels_fcpflags,
+ {"FCP Flags", "fcels.fcpflags", FT_UINT32, BASE_HEX, NULL, 0x0, NULL,
+ HFILL}},
+ { &hf_fcels_fcpflags_trireq,
+ {"Task Retry Ident", "fcels.fcpflags.trireq", FT_BOOLEAN, 32,
+ TFS(&tfs_requested_not_requested), 1 << 9, NULL, HFILL}},
+ { &hf_fcels_fcpflags_trirep,
+ {"Task Retry Ident", "fcels.fcpflags.trirep", FT_BOOLEAN, 32,
+ TFS(&tfs_accepted_not_accepted), 1 << 9, NULL, HFILL}},
+ { &hf_fcels_fcpflags_retry,
+ {"Retry", "fcels.fcpflags.retry", FT_BOOLEAN, 32,
+ TFS(&tfs_fc_fcels_fcpflags_retry), 1 << 8, NULL, HFILL}},
+ { &hf_fcels_fcpflags_ccomp,
+ {"Comp", "fcels.fcpflags.ccomp", FT_BOOLEAN, 32,
+ TFS(&tfs_fc_fcels_fcpflags_ccomp), 0x00000080, NULL, HFILL}},
+ { &hf_fcels_fcpflags_datao,
+ {"Data Overlay", "fcels.fcpflags.datao", FT_BOOLEAN, 32,
+ TFS(&tfs_fc_fcels_fcpflags_datao), 0x00000040, NULL, HFILL}},
+ { &hf_fcels_fcpflags_initiator,
+ {"Initiator", "fcels.fcpflags.initiator", FT_BOOLEAN, 32,
+ TFS(&tfs_fc_fcels_fcpflags_initiator), 0x00000020, NULL, HFILL}},
+ { &hf_fcels_fcpflags_target,
+ {"Target", "fcels.fcpflags.target", FT_BOOLEAN, 32,
+ TFS(&tfs_fc_fcels_fcpflags_target), 0x00000010, NULL, HFILL}},
+ { &hf_fcels_fcpflags_rdxr,
+ {"Rd Xfer_Rdy Dis", "fcels.fcpflags.rdxr", FT_BOOLEAN, 32,
+ TFS(&tfs_fc_fcels_fcpflags_rdxr), 0x00000002, NULL, HFILL}},
+ { &hf_fcels_fcpflags_wrxr,
+ {"Wr Xfer_Rdy Dis", "fcels.fcpflags.wrxr", FT_BOOLEAN, 32,
+ TFS(&tfs_fc_fcels_fcpflags_wrxr), 0x00000001, NULL, HFILL}},
+ { &hf_fcels_prliloflags,
+ {"PRLILO Flags", "fcels.prliloflags", FT_UINT8, BASE_HEX, NULL, 0x0, NULL,
+ HFILL}},
+ { &hf_fcels_tprloflags_opav,
+ {"3rd Party Orig PA Valid", "fcels.tprloflags.opav", FT_BOOLEAN, 8,
+ TFS(&tfs_valid_not_valid), 0x80, NULL, HFILL}},
+ { &hf_fcels_tprloflags_rpav,
+ {"Resp PA Valid", "fcels.tprloflags.rpav", FT_BOOLEAN, 8,
+ TFS(&tfs_valid_not_valid), 0x40, NULL, HFILL}},
+ { &hf_fcels_tprloflags_npv,
+ {"3rd Party N_Port Valid", "fcels.tprloflags.npv", FT_BOOLEAN, 8,
+ TFS(&tfs_valid_not_valid), 0x20, NULL, HFILL}},
+ { &hf_fcels_tprloflags_gprlo,
+ {"Global PRLO", "fcels.tprloflags.gprlo", FT_BOOLEAN, 8,
+ TFS(&tfs_fc_fcels_tprloflags_gprlo), 0x10, NULL, HFILL}},
+ { &hf_fcels_speedflags,
+ {"Port Speed Capabilities", "fcels.speedflags", FT_UINT16, BASE_HEX, NULL, 0x0, NULL,
+ HFILL}},
+ { &hf_fcels_speedflags_1gb,
+ {"1Gb Support", "fcels.speedflags.1gb", FT_BOOLEAN, 16,
+ TFS(&tfs_supported_not_supported), 0x8000, NULL, HFILL}},
+ { &hf_fcels_speedflags_2gb,
+ {"2Gb Support", "fcels.speedflags.2gb", FT_BOOLEAN, 16,
+ TFS(&tfs_supported_not_supported), 0x4000, NULL, HFILL}},
+ { &hf_fcels_speedflags_4gb,
+ {"4Gb Support", "fcels.speedflags.4gb", FT_BOOLEAN, 16,
+ TFS(&tfs_supported_not_supported), 0x2000, NULL, HFILL}},
+ { &hf_fcels_speedflags_10gb,
+ {"10Gb Support", "fcels.speedflags.10gb", FT_BOOLEAN, 16,
+ TFS(&tfs_supported_not_supported), 0x1000, NULL, HFILL}},
+ { &hf_fcels_prliloflags_opav,
+ {"Orig PA Valid", "fcels.prliloflags.opav", FT_BOOLEAN, 8,
+ TFS(&tfs_valid_not_valid), 0x80, NULL, HFILL}},
+ { &hf_fcels_prliloflags_ipe,
+ {"Image Pair Estd", "fcels.prliloflags.ipe", FT_BOOLEAN, 8,
+ TFS(&tfs_fc_fcels_prliloflags_ipe), 0x20, NULL, HFILL}},
+ { &hf_fcels_prliloflags_eip,
+ {"Est Image Pair", "fcels.prliloflags.eip", FT_BOOLEAN, 8,
+ TFS(&tfs_fc_fcels_prliloflags_eip), 0x20, NULL, HFILL}},
+
+ /* Generated from convert_proto_tree_add_text.pl */
+ { &hf_fcels_recovery_qualifier_status, { "Recovery Qualifier Status", "fcels.recovery_qualifier_status", FT_UINT8, BASE_HEX, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_link_failure_count, { "Link Failure Count", "fcels.link_failure_count", FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_loss_of_sync_count, { "Loss of Sync Count", "fcels.loss_of_sync_count", FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_loss_of_signal_count, { "Loss of Signal Count", "fcels.loss_of_signal_count", FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_primitive_seq_protocol_err, { "Primitive Seq Protocol Err", "fcels.primitive_seq_protocol_err", FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_invalid_xmission_word, { "Invalid Xmission Word", "fcels.invalid_xmission_word", FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_invalid_crc_count, { "Invalid CRC Count", "fcels.invalid_crc_count", FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_l_port_status, { "L_Port Status", "fcels.l_port_status", FT_UINT16, BASE_HEX, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_lip_al_ps, { "LIP AL_PS", "fcels.lip.al_ps", FT_UINT8, BASE_HEX, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_lip_f7_initiated_count, { "LIP F7 Initiated Count", "fcels.lip.f7_initiated_count", FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_lip_f7_received_count, { "LIP F7 Received Count", "fcels.lip.f7_received_count", FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_lip_f8_initiated_count, { "LIP F8 Initiated Count", "fcels.lip.f8_initiated_count", FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_lip_f8_received_count, { "LIP F8 Received Count", "fcels.lip.f8_received_count", FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_lip_reset_initiated_count, { "LIP Reset Initiated Count", "fcels.lip.reset_initiated_count", FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_lip_reset_received_count, { "LIP Reset Received Count", "fcels.lip.reset_received_count", FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_rpl_max_size, { "Max Size", "fcels.rpl.max_size", FT_UINT16, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_rpl_index, { "Index", "fcels.rpl.index", FT_UINT24, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_rpl_payload_length, { "Payload Length", "fcels.rpl.payload_length", FT_UINT16, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_rpl_list_length, { "List Length", "fcels.rpl.list_length", FT_UINT24, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_rpl_index_of_i_port_block, { "Index of I Port Block", "fcels.rpl.index_of_i_port_block", FT_UINT24, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_rpl_physical_port, { "Physical Port #", "fcels.rpl.physical_port", FT_UINT32, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_rpl_port_identifier, { "Port Identifier", "fcels.rpl.port_identifier", FT_BYTES, SEP_DOT, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_rpl_port_name, { "Port Name", "fcels.rpl.port_name", FT_FCWWN, BASE_NONE, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_rscn_page_len, { "Page Len", "fcels.rscn.page_len", FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_rscn_payload_len, { "Payload Len", "fcels.rscn.payload_len", FT_UINT16, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_rnft_max_size, { "Max Size", "fcels.rnft.max_size", FT_UINT16, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_rnft_index, { "Index", "fcels.rnft.index", FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_rnft_payload_len, { "Payload Len", "fcels.rnft.payload_len", FT_UINT16, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_rnft_list_length, { "List Length", "fcels.rnft.list_length", FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_rnft_index_of_first_rec_in_list, { "Index of First Rec in List", "fcels.rnft.index_of_first_rec_in_list", FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_rnft_fc4_qualifier, { "FC-4 Qualifier", "fcels.rnft.fc_4_qualifier", FT_UINT24, BASE_HEX, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_prlilo_page_length, { "Page Length", "fcels.prlilo.page_length", FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_prlilo_payload_length, { "Payload Length", "fcels.prlilo.payload_length", FT_UINT16, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_prlilo_type, { "TYPE", "fcels.prlilo.type", FT_UINT8, BASE_DEC, VALS(fc_prli_fc4_val), 0x0, NULL, HFILL }},
+ { &hf_fcels_prlilo_type_code_extension, { "TYPE Code Extension", "fcels.prlilo.type_code_extension", FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_prlilo_response_code, { "Response Code", "fcels.prlilo.response_code", FT_UINT8, BASE_HEX, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_prlilo_originator_pa, { "Originator PA", "fcels.prlilo.originator_pa", FT_UINT32, BASE_HEX, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_prlilo_3rd_party_originator_pa, { "3rd Party Originator PA", "fcels.prlilo.3rd_party_originator_pa", FT_UINT32, BASE_HEX, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_prlilo_responder_pa, { "Responder PA", "fcels.prlilo.responder_pa", FT_UINT32, BASE_HEX, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_prlilo_service_parameter_response, { "Service Parameter Response", "fcels.prlilo.service_parameter_response", FT_UINT32, BASE_HEX, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_prlilo_3rd_party_n_port_id, { "3rd Party N_Port Id", "fcels.prlilo.3rd_party_n_port_id", FT_BYTES, SEP_DOT, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_lirr_regn_function, { "Regn. Function", "fcels.lirr.regn_function", FT_UINT8, BASE_HEX, VALS(fc_els_lirr_regfunc_val), 0x0, NULL, HFILL }},
+ { &hf_fcels_lirr_regn_format, { "Regn. Format", "fcels.lirr.regn_format", FT_UINT8, BASE_HEX, VALS(fc_fc4_val), 0x0, NULL, HFILL }},
+ { &hf_fcels_srl_flag, { "Flag", "fcels.srl.flag", FT_BOOLEAN, 8, TFS(&tfs_srl_flag), 0x01, NULL, HFILL }},
+ { &hf_fcels_srl_fl_port_addr, { "FL_Port Addr", "fcels.srl.fl_port_addr", FT_BYTES, SEP_DOT, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_rpsc_number_of_entries, { "Number of Entries", "fcels.rpsc.number_of_entries", FT_UINT16, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ { &hf_fcels_rpsc_port_oper_speed, { "Port Oper Speed", "fcels.rpsc.port_oper_speed", FT_UINT16, BASE_HEX, VALS(fc_els_portspeed_val), 0x0, NULL, HFILL }},
+ { &hf_fcels_common_identification_data_length, { "Common Identification Data Length", "fcels.common_identification_data_length", FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL }},
+ };
+
+ static gint *ett[] = {
+ &ett_fcels,
+ &ett_fcels_lsrjt,
+ &ett_fcels_acc,
+ &ett_fcels_logi,
+ &ett_fcels_logi_cmnsvc,
+ &ett_fcels_logi_clssvc,
+ &ett_fcels_logo,
+ &ett_fcels_abtx,
+ &ett_fcels_rsi,
+ &ett_fcels_rrq,
+ &ett_fcels_rec,
+ &ett_fcels_prli,
+ &ett_fcels_prli_svcpg,
+ &ett_fcels_adisc,
+ &ett_fcels_farp,
+ &ett_fcels_rps,
+ &ett_fcels_rpl,
+ &ett_fcels_rplpb,
+ &ett_fcels_fan,
+ &ett_fcels_rscn,
+ &ett_fcels_rscn_rec,
+ &ett_fcels_estat,
+ &ett_fcels_scr,
+ &ett_fcels_rnft,
+ &ett_fcels_rnft_fc4,
+ &ett_fcels_lsts,
+ &ett_fcels_rnid,
+ &ett_fcels_rlir,
+ &ett_fcels_lirr,
+ &ett_fcels_srl,
+ &ett_fcels_rpsc,
+ &ett_fcels_cbind,
+ &ett_fcels_cmnfeatures,
+ &ett_fcels_clsflags,
+ &ett_fcels_initctl,
+ &ett_fcels_rcptctl,
+ &ett_fcels_fcpflags,
+ &ett_fcels_prliloflags,
+ &ett_fcels_speedflags,
+ };
+
+
+ static ei_register_info ei[] = {
+ { &ei_fcels_src_unknown, { "fcels.src.type.unknown", PI_PROTOCOL, PI_WARN, "Unknown source address type", EXPFILL }},
+ { &ei_fcels_dst_unknown, { "fcels.dst.type.unknown", PI_PROTOCOL, PI_WARN, "Unknown destination address type", EXPFILL }},
+ { &ei_fcels_no_record_of_exchange, { "fcels.no_record_of_exchange", PI_UNDECODED, PI_WARN, "No record of Exchange. Unable to decode ACC", EXPFILL }},
+ { &ei_fcels_no_record_of_els_req, { "fcels.no_record_of_els_req", PI_UNDECODED, PI_WARN, "No record of ELS Req. Unable to decode ACC", EXPFILL }},
+ };
+
+ expert_module_t* expert_fcels;
+
+ proto_fcels = proto_register_protocol("FC Extended Link Svc", "FC ELS", "fcels");
+
+ proto_register_field_array(proto_fcels, hf, array_length(hf));
+ proto_register_subtree_array(ett, array_length(ett));
+ expert_fcels = expert_register_protocol(proto_fcels);
+ expert_register_field_array(expert_fcels, ei, array_length(ei));
+ fcels_req_hash = wmem_map_new_autoreset(wmem_epan_scope(), wmem_file_scope(), fcels_hash, fcels_equal);
+
+ els_handle = register_dissector("fcels", dissect_fcels, proto_fcels);
+}
+
+void
+proto_reg_handoff_fcels (void)
+{
+ dissector_add_uint("fc.ftype", FC_FTYPE_ELS, els_handle);
+
+ fcsp_handle = find_dissector_add_dependency ("fcsp", proto_fcels);
+}
+
+
+/*
+ * 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:
+ */