diff options
Diffstat (limited to 'epan/dissectors/packet-fcdns.c')
-rw-r--r-- | epan/dissectors/packet-fcdns.c | 1939 |
1 files changed, 1939 insertions, 0 deletions
diff --git a/epan/dissectors/packet-fcdns.c b/epan/dissectors/packet-fcdns.c new file mode 100644 index 0000000000..78c2099ebd --- /dev/null +++ b/epan/dissectors/packet-fcdns.c @@ -0,0 +1,1939 @@ +/* packet-fc-dns.c + * Routines for FC distributed Name Server (dNS) + * Copyright 2001, Dinesh G Dutt <ddutt@andiamo.com> + * + * $Id$ + * + * Ethereal - Network traffic analyzer + * By Gerald Combs <gerald@ethereal.com> + * Copyright 1998 Gerald Combs + * + * Copied from WHATEVER_FILE_YOU_USED (where "WHATEVER_FILE_YOU_USED" + * is a dissector file; if you just copied this from README.developer, + * don't bother with the "Copied from" - you don't even need to put + * in a "Copied from" if you copied an existing dissector, especially + * if the bulk of the code in the new dissector is your code) + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + */ + +#ifdef HAVE_CONFIG_H +# include "config.h" +#endif + +#include <stdio.h> +#include <stdlib.h> +#include <string.h> + +#ifdef HAVE_SYS_TYPES_H +# include <sys/types.h> +#endif + +#ifdef HAVE_NETINET_IN_H +# include <netinet/in.h> +#endif + +#include <glib.h> + +#ifdef NEED_SNPRINTF_H +# include "snprintf.h" +#endif + +#include <epan/packet.h> +#include <epan/conversation.h> +#include "etypes.h" +#include "packet-fc.h" +#include "packet-fcct.h" +#include "packet-fcdns.h" +#include "packet-fcswils.h" + +/* Initialize the protocol and registered fields */ +static int proto_fcdns = -1; +static int hf_fcdns_gssubtype = -1; +static int hf_fcdns_opcode = -1; +static int hf_fcdns_reason = -1; +static int hf_fcdns_vendor = -1; +static int hf_fcdns_req_portid = -1; +static int hf_fcdns_rply_pname = -1; +static int hf_fcdns_rply_nname = -1; +static int hf_fcdns_rply_cos = -1; +static int hf_fcdns_rply_gft = -1; +static int hf_fcdns_rply_snamelen = -1; +static int hf_fcdns_rply_sname = -1; +static int hf_fcdns_rply_ptype = -1; +static int hf_fcdns_rply_fpname = -1; +static int hf_fcdns_fc4type = -1; +static int hf_fcdns_rply_fc4type = -1; +static int hf_fcdns_rply_fc4desc = -1; +static int hf_fcdns_rply_fc4feat = -1; +static int hf_fcdns_req_pname = -1; +static int hf_fcdns_rply_portid = -1; +static int hf_fcdns_req_nname = -1; +static int hf_fcdns_req_domainscope = -1; +static int hf_fcdns_req_areascope = -1; +static int hf_fcdns_req_fc4type = -1; +static int hf_fcdns_req_ptype = -1; +static int hf_fcdns_req_fc4feature = -1; +static int hf_fcdns_req_cos = -1; +static int hf_fcdns_req_fc4types = -1; +static int hf_fcdns_req_snamelen = -1; +static int hf_fcdns_req_sname = -1; +static int hf_fcdns_rply_spnamelen = -1; +static int hf_fcdns_rply_spname = -1; +static int hf_fcdns_req_spnamelen = -1; +static int hf_fcdns_req_spname = -1; +static int hf_fcdns_rply_ipa = -1; +static int hf_fcdns_rply_ipnode = -1; +static int hf_fcdns_rply_ipport = -1; +static int hf_fcdns_rply_fc4desclen = -1; +static int hf_fcdns_rply_hrdaddr = -1; +static int hf_fcdns_req_fdesclen = -1; +static int hf_fcdns_req_fdesc = -1; +static int hf_fcdns_req_ip = -1; +static int hf_fcdns_rjtdetail = -1; +static int hf_fcdns_zone_mbrtype = -1; +static int hf_fcdns_zone_mbrid = -1; +static int hf_fcdns_zonenm = -1; +static int hf_fcdns_portip = -1; +static int hf_fcdns_sw2_objfmt = -1; +static int hf_fcdns_num_fc4desc = -1; +static int hf_fcdns_rply_ownerid = -1; +static int hf_fcdns_maxres_size = -1; + + +/* Initialize the subtree pointers */ +static gint ett_fcdns = -1; + +typedef struct _fcdns_conv_key { + guint32 conv_idx; +} fcdns_conv_key_t; + +typedef struct _fcdns_conv_data { + guint32 opcode; +} fcdns_conv_data_t; + +GHashTable *fcdns_req_hash = NULL; +GMemChunk *fcdns_req_keys = NULL; +GMemChunk *fcdns_req_vals = NULL; +guint32 fcdns_init_count = 25; + +static dissector_handle_t data_handle; + +/* + * Hash Functions + */ +static gint +fcdns_equal(gconstpointer v, gconstpointer w) +{ + fcdns_conv_key_t *v1 = (fcdns_conv_key_t *)v; + fcdns_conv_key_t *v2 = (fcdns_conv_key_t *)w; + + return (v1->conv_idx == v2->conv_idx); +} + +static guint +fcdns_hash (gconstpointer v) +{ + fcdns_conv_key_t *key = (fcdns_conv_key_t *)v; + guint val; + + val = key->conv_idx; + + return val; +} + +/* + * Protocol initialization + */ +static void +fcdns_init_protocol(void) +{ + if (fcdns_req_keys) + g_mem_chunk_destroy (fcdns_req_keys); + if (fcdns_req_vals) + g_mem_chunk_destroy (fcdns_req_vals); + if (fcdns_req_hash) + g_hash_table_destroy(fcdns_req_hash); + + fcdns_req_hash = g_hash_table_new(fcdns_hash, fcdns_equal); + fcdns_req_keys = g_mem_chunk_new ("fcdns_req_keys", + sizeof(fcdns_conv_key_t), + fcdns_init_count * + sizeof(fcdns_conv_key_t), + G_ALLOC_AND_FREE); + fcdns_req_vals = g_mem_chunk_new ("fcdns_req_vals", + sizeof(fcdns_conv_data_t), + fcdns_init_count * + sizeof(fcdns_conv_data_t), + G_ALLOC_AND_FREE); +} + +static gchar * +fccos_to_str (tvbuff_t *tvb, int offset, gchar *cosstr) +{ + int stroff = 0, + cos = 0; + + if (cosstr == NULL) + return NULL; + + cos = tvb_get_ntohl (tvb, offset); + + cosstr[0] = '\0'; + + if (cos & 0x1) { + strcpy (cosstr, "F, "); + stroff += 3; + } + + if (cos & 0x2) { + strcpy (&cosstr[stroff], "1, "); + stroff += 3; + } + + if (cos & 0x4) { + strcpy (&cosstr[stroff], "2, "); + stroff += 3; + } + + if (cos & 0x8) { + strcpy (&cosstr[stroff], "3, "); + stroff += 3; + } + + if (cos & 0x10) { + strcpy (&cosstr[stroff], "4, "); + stroff += 3; + } + + if (cos & 0x40) { + strcpy (&cosstr[stroff], "6"); + } + + return (cosstr); +} + +/* The feature routines just decode FCP's FC-4 features field */ +static gchar * +fc4feature_to_str (guint8 fc4feature, guint8 fc4type, gchar *str) +{ + int stroff = 0; + + *str = '\0'; + + if (fc4type == FC_TYPE_SCSI) { + if (fc4feature & 0x1) { + strcpy (str, "T, "); + stroff += 3; + } + + if (fc4feature & 0x2) { + strcpy (&str[stroff], "I"); + } + } + else { + sprintf (str, "0x%x", fc4feature); + } + return (str); +} + +static gchar * +fc4ftrs_to_str (tvbuff_t *tvb, int offset, gchar *str) +{ + guint8 fc4feature; + int stroff = 0; + + if (str == NULL) { + return NULL; + } + + *str = '\0'; + fc4feature = tvb_get_guint8 (tvb, offset+7); + + if (fc4feature & 0x1) { + strcpy (str, "T, "); + stroff += 3; + } + + if (fc4feature & 0x2) { + strcpy (&str[stroff], "I"); + } + + return (str); +} + +/* Decodes LLC/SNAP, IP, FCP, VI, GS, SW_ILS types only */ +/* Max len of str to be allocated by caller is 40 */ +static gchar * +fc4type_to_str (tvbuff_t *tvb, int offset, gchar *str) +{ + guint32 fc4tword; + int stroff = 0; + + if (str == NULL) { + return NULL; + } + + *str = '\0'; + + fc4tword = tvb_get_ntohl (tvb, offset); + + if (fc4tword & 0x10) { + strcpy (str, "LLC/SNAP, "); + stroff += 10; + } + + if (fc4tword & 0x20) { + strcpy (&str[stroff], "IP, "); + stroff += 4; + } + + if (fc4tword & 0x0100) { + strcpy (&str[stroff], "FCP, "); + stroff += 5; + } + + fc4tword = tvb_get_ntohl (tvb, offset+4); + + if (fc4tword & 0x1) { + strcpy (&str[stroff], "GS3, "); + stroff += 5; + } + + if (fc4tword & 0x4) { + strcpy (&str[stroff], "SNMP, "); + stroff += 6; + } + + if (fc4tword & 0x10) { + strcpy (&str[stroff], "SW_ILS, "); + stroff += 8; + } + + fc4tword = tvb_get_ntohl (tvb, offset+8); + if (fc4tword & 0x1) { + strcpy (&str[stroff], "VI, "); + stroff += 3; + } + return (str); +} + +/* Code to actually dissect the packets */ + +/* A bunch of get routines have a similar req packet format. The first few + * routines deal with this decoding. All assume that tree is valid */ +static void +dissect_fcdns_req_portid (tvbuff_t *tvb, proto_tree *tree, int offset) +{ + if (tree) { + proto_tree_add_string (tree, hf_fcdns_req_portid, tvb, offset, 3, + fc_to_str (tvb_get_ptr (tvb, offset, 3))); + } +} + +static void +dissect_fcdns_ganxt (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + guint8 len; + gchar str[128]; + + if (req_tree) { + if (isreq) { + dissect_fcdns_req_portid (tvb, req_tree, offset+1); + } + else { + proto_tree_add_item (req_tree, hf_fcdns_rply_ptype, tvb, offset, + 1, 0); + proto_tree_add_string (req_tree, hf_fcdns_rply_portid, tvb, + offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, 3))); + proto_tree_add_string (req_tree, hf_fcdns_rply_pname, tvb, + offset+4, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset+4, + 8))); + len = tvb_get_guint8 (tvb, offset+12); + proto_tree_add_item (req_tree, hf_fcdns_rply_spnamelen, tvb, + offset+12, 1, 0); + if (!tvb_offset_exists (tvb, 29+len)) + return; + + if (len) { + proto_tree_add_item (req_tree, hf_fcdns_rply_spname, tvb, + offset+13, len, 0); + } + + if (tvb_offset_exists (tvb, 292)) { + proto_tree_add_string (req_tree, hf_fcdns_rply_nname, tvb, + offset+268, 8, + fcwwn_to_str (tvb_get_ptr (tvb, + offset+268, + 0))); + } + if (tvb_offset_exists (tvb, 548)) { + len = tvb_get_guint8 (tvb, offset+276); + proto_tree_add_item (req_tree, hf_fcdns_rply_snamelen, tvb, + offset+276, 1, 0); + if (len) { + proto_tree_add_item (req_tree, hf_fcdns_rply_sname, tvb, + offset+277, len, 0); + } + } + if (tvb_offset_exists (tvb, 556)) { + proto_tree_add_item (req_tree, hf_fcdns_rply_ipa, tvb, + offset+532, 8, 0); + } + if (tvb_offset_exists (tvb, 572)) { + proto_tree_add_item (req_tree, hf_fcdns_rply_ipnode, tvb, + offset+540, 16, 0); + } + if (tvb_offset_exists (tvb, 576)) { + proto_tree_add_string (req_tree, hf_fcdns_rply_cos, tvb, offset+556, + 4, + fccos_to_str (tvb, offset+556, str)); + } + if (tvb_offset_exists (tvb, 608)) { + proto_tree_add_string (req_tree, hf_fcdns_rply_gft, tvb, offset+560, + 32, + fc4type_to_str (tvb, offset+560, str)); + } + if (tvb_offset_exists (tvb, 624)) { + proto_tree_add_item (req_tree, hf_fcdns_rply_ipport, tvb, + offset+592, 16, 0); + } + if (tvb_offset_exists (tvb, 632)) { + proto_tree_add_string (req_tree, hf_fcdns_rply_fpname, tvb, + offset+608, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset+608, + 8))); + } + if (tvb_offset_exists (tvb, 635)) { + proto_tree_add_string (req_tree, hf_fcdns_rply_hrdaddr, tvb, + offset+617, 3, + fc_to_str (tvb_get_ptr (tvb, offset+617, + 3))); + } + } + } +} + +static void +dissect_fcdns_gpnid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (req_tree) { + if (isreq) { + dissect_fcdns_req_portid (tvb, req_tree, offset+1); + } + else { + proto_tree_add_string (req_tree, hf_fcdns_rply_pname, tvb, offset, + 8, fcwwn_to_str (tvb_get_ptr (tvb, offset, + 8))); + } + } +} + +static void +dissect_fcdns_gnnid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (req_tree) { + if (isreq) { + dissect_fcdns_req_portid (tvb, req_tree, offset+1); + } + else { + proto_tree_add_string (req_tree, hf_fcdns_rply_nname, tvb, + offset, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset, + 8))); + } + } +} + +static void +dissect_fcdns_gcsid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + gchar cosstr[64]; + + if (req_tree) { + if (isreq) { + dissect_fcdns_req_portid (tvb, req_tree, offset); + } + else { + proto_tree_add_string (req_tree, hf_fcdns_rply_cos, tvb, + offset, 4, + fccos_to_str (tvb, offset, cosstr)); + } + } +} + +static void +dissect_fcdns_gftid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + gchar fc4str[64]; + + if (req_tree) { + if (isreq) { + dissect_fcdns_req_portid (tvb, req_tree, offset+1); + } + else { + proto_tree_add_string (req_tree, hf_fcdns_rply_gft, tvb, + offset, 32, + fc4type_to_str (tvb, offset, fc4str)); + } + } +} + +static void +dissect_fcdns_gspnid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + guint8 len; + + if (req_tree) { + if (isreq) { + dissect_fcdns_req_portid (tvb, req_tree, offset+1); + } + else { + len = tvb_get_guint8 (tvb, offset); + proto_tree_add_item (req_tree, hf_fcdns_rply_spnamelen, + tvb, offset, 1, 0); + proto_tree_add_string (req_tree, hf_fcdns_rply_spname, tvb, + offset+1, len, + tvb_get_ptr (tvb, offset+1, len)); + } + } +} + +static void +dissect_fcdns_gptid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (req_tree) { + if (isreq) { + dissect_fcdns_req_portid (tvb, req_tree, offset+1); + } + else { + proto_tree_add_item (req_tree, hf_fcdns_rply_ptype, tvb, + offset, 1, 0); + } + } +} + +static void +dissect_fcdns_gfpnid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (req_tree) { + if (isreq) { + dissect_fcdns_req_portid (tvb, req_tree, offset+1); + } + else { + proto_tree_add_string (req_tree, hf_fcdns_rply_fpname, tvb, + offset, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset, + 8))); + } + } + +} + +static void +dissect_fcdns_gfdid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + gchar fc4str[128]; + int tot_len, desclen; + + if (req_tree) { + if (isreq) { + dissect_fcdns_req_portid (tvb, req_tree, offset+1); + proto_tree_add_string (req_tree, hf_fcdns_fc4type, tvb, offset+4, + 32, + fc4type_to_str (tvb, offset+4, fc4str)); + } + else { + tot_len = tvb_length (tvb) - offset; /* excluding CT header */ + while (tot_len > 0) { + /* The count of the descriptors is not returned and so we have + * to track the display by the length field */ + desclen = tvb_get_guint8 (tvb, offset); + proto_tree_add_item (req_tree, hf_fcdns_rply_fc4desc, tvb, + offset, desclen, 0); + tot_len -= 255; /* descriptors are aligned to 255 bytes */ + offset += 256; + } + } + } +} + +static void +dissect_fcdns_gffid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + gchar fc4fstr[128]; + + if (req_tree) { + if (isreq) { + dissect_fcdns_req_portid (tvb, req_tree, offset+1); + } + else { + proto_tree_add_string (req_tree, hf_fcdns_rply_fc4feat, tvb, + offset, 128, + fc4ftrs_to_str (tvb, offset, fc4fstr)); + } + } +} + +static void +dissect_fcdns_gidpn (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (req_tree) { + if (isreq) { + proto_tree_add_string (req_tree, hf_fcdns_req_pname, tvb, + offset, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset, + 8))); + } + else { + proto_tree_add_string (req_tree, hf_fcdns_rply_portid, tvb, + offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + } + } +} + +static void +dissect_fcdns_gipppn (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (req_tree) { + if (isreq) { + proto_tree_add_string (req_tree, hf_fcdns_req_pname, tvb, + offset, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset, + 8))); + } + else { + proto_tree_add_item (req_tree, hf_fcdns_rply_ipport, tvb, offset, + 16, 0); + } + } +} + +static void +dissect_fcdns_gidnn (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + guint8 islast; + + if (req_tree) { + if (isreq) { + proto_tree_add_string (req_tree, hf_fcdns_req_nname, tvb, + offset, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset, + 8))); + } + else { + do { + islast = tvb_get_guint8 (tvb, offset); + proto_tree_add_string (req_tree, hf_fcdns_rply_portid, + tvb, offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + offset += 4; + } while (!(islast & 0x80)); + } + } +} + +static void +dissect_fcdns_gipnn (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (req_tree) { + if (isreq) { + proto_tree_add_string (req_tree, hf_fcdns_req_nname, tvb, + offset, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset, + 8))); + } + else { + proto_tree_add_item (req_tree, hf_fcdns_rply_ipnode, tvb, offset, + 16, 0); + } + } +} + +static void +dissect_fcdns_gpnnn (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + guint8 islast; + + if (req_tree) { + if (isreq) { + proto_tree_add_string (req_tree, hf_fcdns_req_nname, tvb, + offset, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset, + 8))); + } + else { + do { + islast = tvb_get_guint8 (tvb, offset); + proto_tree_add_string (req_tree, hf_fcdns_rply_portid, + tvb, offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + proto_tree_add_string (req_tree, hf_fcdns_rply_pname, + tvb, offset+8, 8, + fcwwn_to_str (tvb_get_ptr (tvb, + offset+8, + 8))); + offset += 16; + } while (!(islast & 0x80)); + } + } +} + +static void +dissect_fcdns_gsnnnn (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + guint8 len; + + if (req_tree) { + if (isreq) { + proto_tree_add_string (req_tree, hf_fcdns_req_nname, tvb, + offset, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset, + 8))); + } + else { + len = tvb_get_guint8 (tvb, offset); + proto_tree_add_item (req_tree, hf_fcdns_rply_snamelen, tvb, + offset, 1, 0); + proto_tree_add_item (req_tree, hf_fcdns_rply_sname, tvb, + offset+1, len, 0); + } + } +} + +static void +dissect_fcdns_gidft (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + guint8 islast; + + if (req_tree) { + if (isreq) { + proto_tree_add_item (req_tree, hf_fcdns_req_domainscope, + tvb, offset+1, 1, 0); + proto_tree_add_item (req_tree, hf_fcdns_req_areascope, + tvb, offset+2, 1, 0); + proto_tree_add_item (req_tree, hf_fcdns_req_fc4type, + tvb, offset+3, 1, 0); + } + else { + do { + islast = tvb_get_guint8 (tvb, offset); + proto_tree_add_string (req_tree, hf_fcdns_rply_portid, + tvb, offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + offset += 4; + } while (!(islast & 0x80)); + } + } +} + +static void +dissect_fcdns_gpnft (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + guint8 islast; + + if (req_tree) { + if (isreq) { + proto_tree_add_item (req_tree, hf_fcdns_req_domainscope, + tvb, offset+1, 1, 0); + proto_tree_add_item (req_tree, hf_fcdns_req_areascope, + tvb, offset+2, 1, 0); + proto_tree_add_item (req_tree, hf_fcdns_req_fc4type, + tvb, offset+3, 1, 0); + } + else { + do { + islast = tvb_get_guint8 (tvb, offset); + proto_tree_add_string (req_tree, hf_fcdns_rply_portid, + tvb, offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + proto_tree_add_string (req_tree, hf_fcdns_rply_pname, + tvb, offset+4, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset+8, + 8))); + offset += 16; + } while (!(islast & 0x80)); + } + } +} + +static void +dissect_fcdns_gnnft (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + guint8 islast; + + if (req_tree) { + if (isreq) { + proto_tree_add_item (req_tree, hf_fcdns_req_domainscope, + tvb, offset+1, 1, 0); + proto_tree_add_item (req_tree, hf_fcdns_req_areascope, + tvb, offset+2, 1, 0); + proto_tree_add_item (req_tree, hf_fcdns_req_fc4type, + tvb, offset+3, 1, 0); + } + else { + do { + islast = tvb_get_guint8 (tvb, offset); + proto_tree_add_string (req_tree, hf_fcdns_rply_portid, + tvb, offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + proto_tree_add_string (req_tree, hf_fcdns_rply_nname, + tvb, offset+4, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset+8, + 8))); + offset += 16; + } while (!(islast & 0x80)); + } + } +} + +static void +dissect_fcdns_gidpt (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + guint8 islast = 0; + + if (req_tree) { + if (isreq) { + proto_tree_add_item (req_tree, hf_fcdns_req_ptype, + tvb, offset, 1, 0); + proto_tree_add_item (req_tree, hf_fcdns_req_domainscope, + tvb, offset+1, 1, 0); + proto_tree_add_item (req_tree, hf_fcdns_req_areascope, + tvb, offset+2, 1, 0); + } + else { + do { + islast = tvb_get_guint8 (tvb, offset); + proto_tree_add_string (req_tree, hf_fcdns_rply_portid, + tvb, offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + offset += 4; + } while (!(islast & 0x80)); + } + } +} + +static void +dissect_fcdns_gidipp (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + guint8 islast; + + if (req_tree) { + if (isreq) { + proto_tree_add_item (req_tree, hf_fcdns_req_ip, tvb, offset, + 16, 0); + } + else { + do { + islast = tvb_get_guint8 (tvb, offset); + proto_tree_add_string (req_tree, hf_fcdns_rply_portid, + tvb, offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + offset += 4; + } while (!(islast & 0x80)); + } + } +} + +static void +dissect_fcdns_gidff (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + guint8 islast; + gchar str[64]; + + if (req_tree) { + if (isreq) { + proto_tree_add_item (req_tree, hf_fcdns_req_domainscope, tvb, + offset+1, 1, 0); + proto_tree_add_item (req_tree, hf_fcdns_req_areascope, tvb, + offset+2, 1, 0); + proto_tree_add_string (req_tree, hf_fcdns_req_fc4feature, tvb, + offset+6, 1, + fc4feature_to_str (tvb_get_guint8 (tvb, offset+6), + tvb_get_guint8 (tvb, offset+7), + str)); + proto_tree_add_item (req_tree, hf_fcdns_req_fc4type, tvb, + offset+7, 1, 0); + } + else { + do { + islast = tvb_get_guint8 (tvb, offset); + proto_tree_add_string (req_tree, hf_fcdns_rply_portid, + tvb, offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + offset += 4; + } while (!(islast & 0x80)); + } + } +} + +static void +dissect_fcdns_rpnid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (req_tree) { + if (isreq) { + proto_tree_add_string (req_tree, hf_fcdns_req_portid, + tvb, offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + proto_tree_add_string (req_tree, hf_fcdns_req_pname, tvb, + offset+4, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset+4, + 8))); + } + } +} + +static void +dissect_fcdns_rnnid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (req_tree) { + if (isreq) { + proto_tree_add_string (req_tree, hf_fcdns_req_portid, + tvb, offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + proto_tree_add_string (req_tree, hf_fcdns_req_nname, tvb, + offset+4, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset+4, + 8))); + } + } +} + +static void +dissect_fcdns_rcsid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + gchar cos[64]; + + if (req_tree && isreq) { + proto_tree_add_string (req_tree, hf_fcdns_req_portid, tvb, + offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + proto_tree_add_string (req_tree, hf_fcdns_req_cos, tvb, + offset+4, 4, + fccos_to_str (tvb, offset+4, cos)); + } +} + +static void +dissect_fcdns_rptid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (req_tree && isreq) { + proto_tree_add_string (req_tree, hf_fcdns_req_portid, tvb, + offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + proto_tree_add_item (req_tree, hf_fcdns_req_ptype, tvb, + offset+4, 1, 0); + } +} + +static void +dissect_fcdns_rftid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + gchar fc4str[128]; + + if (req_tree && isreq) { + proto_tree_add_string (req_tree, hf_fcdns_req_portid, tvb, + offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + proto_tree_add_string (req_tree, hf_fcdns_req_fc4types, tvb, + offset+4, 32, + fc4type_to_str (tvb, offset+4, fc4str)); + } +} + +static void +dissect_fcdns_rspnid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + guint8 len; + + if (req_tree && isreq) { + proto_tree_add_string (req_tree, hf_fcdns_req_portid, tvb, + offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + proto_tree_add_item (req_tree, hf_fcdns_req_spnamelen, tvb, + offset+4, 1, 0); + len = tvb_get_guint8 (tvb, offset+4); + + proto_tree_add_item (req_tree, hf_fcdns_req_spname, tvb, offset+5, + len, 0); + } +} + +static void +dissect_fcdns_rippid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (req_tree && isreq) { + proto_tree_add_string (req_tree, hf_fcdns_req_portid, tvb, + offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + proto_tree_add_item (req_tree, hf_fcdns_req_ip, tvb, + offset+4, 16, 0); + } +} + +static void +dissect_fcdns_rfdid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + int len, dlen; + gchar fc4str[128]; + + if (req_tree && isreq) { + proto_tree_add_string (req_tree, hf_fcdns_req_portid, tvb, + offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + proto_tree_add_string (req_tree, hf_fcdns_req_fc4types, tvb, + offset+4, 32, + fc4type_to_str (tvb, offset+4, fc4str)); + + len = tvb_length (tvb) - offset - 36; + offset += 36; + + while (len > 0) { + dlen = tvb_get_guint8 (tvb, offset); + proto_tree_add_item (req_tree, hf_fcdns_req_fdesclen, tvb, offset, + 1, 0); + proto_tree_add_item (req_tree, hf_fcdns_req_fdesc, tvb, offset+1, + len, 0); + offset += 256; + len -= 256; + } + } +} + +static void +dissect_fcdns_rffid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + gchar fc4str[64]; + + if (req_tree && isreq) { + proto_tree_add_string (req_tree, hf_fcdns_req_portid, tvb, offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, 3))); + proto_tree_add_string (req_tree, hf_fcdns_req_fc4feature, tvb, + offset+6, 1, + fc4feature_to_str (tvb_get_guint8 (tvb, + offset+6), + tvb_get_guint8 (tvb, + offset+7), + fc4str)); + proto_tree_add_item (req_tree, hf_fcdns_req_fc4type, tvb, offset+7, + 1, 0); + } +} + +static void +dissect_fcdns_ripnn (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (req_tree && isreq) { + proto_tree_add_string (req_tree, hf_fcdns_req_nname, tvb, offset, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset, 8))); + proto_tree_add_item (req_tree, hf_fcdns_req_ip, tvb, offset+8, 16, 0); + } +} + +static void +dissect_fcdns_rsnnnn (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + guint8 len; + + if (req_tree && isreq) { + proto_tree_add_string (req_tree, hf_fcdns_req_nname, tvb, offset, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset, 8))); + len = tvb_get_guint8 (tvb, offset+8); + + proto_tree_add_item (req_tree, hf_fcdns_req_snamelen, tvb, offset+8, + 1, 0); + proto_tree_add_item (req_tree, hf_fcdns_req_sname, tvb, offset+9, + len, 0); + } +} + +static void +dissect_fcdns_daid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (req_tree && isreq) { + proto_tree_add_string (req_tree, hf_fcdns_req_portid, tvb, offset+1, 3, + fc_to_str (tvb_get_ptr (tvb, offset+1, 3))); + } +} + +static gchar * +zonenm_to_str (tvbuff_t *tvb, gint offset) +{ + int len = tvb_get_guint8 (tvb, offset); + return ((gchar *)tvb_get_ptr (tvb, offset+4, len)); +} + +static void +dissect_fcdns_zone_mbr (tvbuff_t *tvb, proto_tree *zmbr_tree, int offset) +{ + int mbrlen = 4 + tvb_get_guint8 (tvb, offset+3); + + proto_tree_add_item (zmbr_tree, hf_fcdns_zone_mbrtype, tvb, + offset, 1, 0); + proto_tree_add_text (zmbr_tree, tvb, offset+2, 1, "Flags: 0x%x", + tvb_get_guint8 (tvb, offset+2)); + proto_tree_add_text (zmbr_tree, tvb, offset+3, 1, + "Identifier Length: %d", + tvb_get_guint8 (tvb, offset+3)); + switch (tvb_get_guint8 (tvb, offset)) { + case FC_SWILS_ZONEMBR_WWN: + proto_tree_add_string (zmbr_tree, hf_fcdns_zone_mbrid, tvb, + offset+4, 8, + fcwwn_to_str (tvb_get_ptr (tvb, + offset+4, + 8))); + break; + case FC_SWILS_ZONEMBR_DP: + proto_tree_add_string_format (zmbr_tree, + hf_fcdns_zone_mbrid, + tvb, offset+4, 4, " ", + "0x%x", + tvb_get_ntohl (tvb, + offset+4)); + break; + case FC_SWILS_ZONEMBR_FCID: + proto_tree_add_string (zmbr_tree, hf_fcdns_zone_mbrid, tvb, + offset+4, 4, + fc_to_str (tvb_get_ptr (tvb, + offset+4, + 3))); + break; + case FC_SWILS_ZONEMBR_ALIAS: + proto_tree_add_string (zmbr_tree, hf_fcdns_zone_mbrid, tvb, + offset+4, + tvb_get_guint8 (tvb, offset+3), + zonenm_to_str (tvb, offset+4)); + break; + default: + proto_tree_add_string (zmbr_tree, hf_fcdns_zone_mbrid, tvb, + offset+4, mbrlen, + "Unknown member type format"); + + } +} + +static void +dissect_fcdns_swils_entries (tvbuff_t *tvb, proto_tree *tree, int offset) +{ + int numrec, i, len; + guint8 objfmt; + gchar str[512]; + + numrec = tvb_get_ntohl (tvb, offset); + + if (tree) { + proto_tree_add_text (tree, tvb, offset, 4, "Number of Entries: %d", + numrec); + offset += 4; + + for (i = 0; i < numrec; i++) { + objfmt = tvb_get_guint8 (tvb, offset); + + proto_tree_add_item (tree, hf_fcdns_sw2_objfmt, tvb, offset, 1, 0); + proto_tree_add_string (tree, hf_fcdns_rply_ownerid, tvb, offset+1, + 3, fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + proto_tree_add_item (tree, hf_fcdns_rply_ptype, tvb, offset+4, + 1, 0); + proto_tree_add_string (tree, hf_fcdns_rply_portid, tvb, offset+5, 3, + fc_to_str (tvb_get_ptr (tvb, offset+5, 3))); + proto_tree_add_string (tree, hf_fcdns_rply_pname, tvb, offset+8, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset+8, + 8))); + offset += 16; + if (!(objfmt & 0x1)) { + len = tvb_get_guint8 (tvb, offset); + proto_tree_add_item (tree, hf_fcdns_rply_spnamelen, tvb, + offset, 1, 0); + proto_tree_add_item (tree, hf_fcdns_rply_spname, tvb, + offset+1, len, 0); + offset += 256; + } + proto_tree_add_string (tree, hf_fcdns_rply_nname, tvb, offset, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset, 8))); + offset += 8; + if (!(objfmt & 0x1)) { + len = tvb_get_guint8 (tvb, offset); + proto_tree_add_item (tree, hf_fcdns_rply_snamelen, tvb, + offset, 1, 0); + proto_tree_add_item (tree, hf_fcdns_rply_sname, tvb, + offset+1, len, 0); + offset += 256; + } + proto_tree_add_item (tree, hf_fcdns_rply_ipa, tvb, offset, 8, 0); + proto_tree_add_item (tree, hf_fcdns_rply_ipnode, tvb, offset+8, 16, + 0); + proto_tree_add_string (tree, hf_fcdns_rply_cos, tvb, offset+24, 4, + fccos_to_str (tvb, offset+24, str)); + proto_tree_add_string (tree, hf_fcdns_rply_gft, tvb, offset+28, + 32, + fc4type_to_str (tvb, offset+28, str)); + proto_tree_add_item (tree, hf_fcdns_rply_ipport, tvb, offset+60, + 16, 0); + proto_tree_add_string (tree, hf_fcdns_rply_fpname, tvb, offset+76, + 8, fcwwn_to_str (tvb_get_ptr (tvb, + offset+76, + 8))); + proto_tree_add_string (tree, hf_fcdns_rply_hrdaddr, tvb, offset+85, + 3, fc_to_str (tvb_get_ptr (tvb, offset+85, + 3))); + offset += 88; + if (objfmt & 0x2) { + proto_tree_add_string (tree, hf_fcdns_rply_fc4feat, tvb, + offset, 128, + fc4ftrs_to_str (tvb, offset, str)); + if (tvb_get_guint8 (tvb, offset+129)) { + proto_tree_add_item (tree, hf_fcdns_rply_fc4type, tvb, + offset+128, 1, 0); + proto_tree_add_item (tree, hf_fcdns_num_fc4desc, tvb, + offset+129, 1, 0); + len = tvb_get_guint8 (tvb, offset+132); + proto_tree_add_item (tree, hf_fcdns_rply_fc4desclen, tvb, + offset+132, 1, 0); + proto_tree_add_item (tree, hf_fcdns_rply_fc4desc, tvb, + offset+133, len, 0); + } + else { + proto_tree_add_item (tree, hf_fcdns_num_fc4desc, tvb, + offset+129, 1, 0); + } + offset += 388; /* FC4 desc is 260 bytes, maybe padded */ + } + } + } +} + +static void +dissect_fcdns_geid (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (isreq) { + if (req_tree) { + proto_tree_add_string (req_tree, hf_fcdns_req_portid, tvb, offset+1, + 3, fc_to_str (tvb_get_ptr (tvb, offset+1, + 3))); + } + } + else { + dissect_fcdns_swils_entries (tvb, req_tree, offset); + } +} + +static void +dissect_fcdns_gepn (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + if (isreq) { + if (req_tree) { + proto_tree_add_string (req_tree, hf_fcdns_req_pname, tvb, offset, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset, 8))); + } + } + else { + dissect_fcdns_swils_entries (tvb, req_tree, offset); + } +} + +static void +dissect_fcdns_genn (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (isreq) { + if (req_tree) { + proto_tree_add_string (req_tree, hf_fcdns_req_nname, tvb, offset, 8, + fcwwn_to_str (tvb_get_ptr (tvb, offset, 8))); + } + } + else { + dissect_fcdns_swils_entries (tvb, req_tree, offset); + } +} + +static void +dissect_fcdns_geip (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (isreq) { + if (req_tree) { + proto_tree_add_item (req_tree, hf_fcdns_req_ip, tvb, offset, 16, 0); + } + } + else { + dissect_fcdns_swils_entries (tvb, req_tree, offset); + } +} + +static void +dissect_fcdns_geft (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + gchar str[128]; + + if (isreq) { + if (req_tree) { + proto_tree_add_string (req_tree, hf_fcdns_fc4type, tvb, offset, 32, + fc4type_to_str (tvb, offset, str)); + } + } + else { + dissect_fcdns_swils_entries (tvb, req_tree, offset); + } +} + +static void +dissect_fcdns_gept (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (isreq) { + if (req_tree) { + proto_tree_add_item (req_tree, hf_fcdns_req_ptype, tvb, offset+3, + 1, 0); + } + } + else { + dissect_fcdns_swils_entries (tvb, req_tree, offset); + } +} + +static void +dissect_fcdns_gezm (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (isreq) { + if (req_tree) { + dissect_fcdns_zone_mbr (tvb, req_tree, offset); + } + } + else { + dissect_fcdns_swils_entries (tvb, req_tree, offset); + } +} + +static void +dissect_fcdns_gezn (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + int strlen; + + if (isreq) { + if (req_tree) { + strlen = tvb_get_guint8 (tvb, offset); + proto_tree_add_text (req_tree, tvb, offset, 1, "Name Length: %d", + strlen); + proto_tree_add_string (req_tree, hf_fcdns_zonenm, tvb, offset+3, + strlen, tvb_get_ptr (tvb, offset+3, strlen)); + } + } + else { + dissect_fcdns_swils_entries (tvb, req_tree, offset); + } +} + +static void +dissect_fcdns_geipp (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + + if (isreq) { + if (req_tree) { + proto_tree_add_item (req_tree, hf_fcdns_portip, tvb, offset, 4, 0); + } + } + else { + dissect_fcdns_swils_entries (tvb, req_tree, offset); + } +} + +static void +dissect_fcdns_geff (tvbuff_t *tvb, proto_tree *req_tree, gboolean isreq) +{ + int offset = 16; /* past the fc_ct header */ + gchar str[256]; + + if (isreq) { + if (req_tree) { + proto_tree_add_string (req_tree, hf_fcdns_req_fc4feature, tvb, offset, + 128, fc4ftrs_to_str (tvb, offset, str)); + } + } + else { + dissect_fcdns_swils_entries (tvb, req_tree, offset); + } +} + +static void +dissect_fcdns_rjt (tvbuff_t *tvb, proto_tree *req_tree) +{ + int offset = 0; + + if (req_tree) { + proto_tree_add_item (req_tree, hf_fcdns_reason, tvb, offset+13, 1, 0); + proto_tree_add_item (req_tree, hf_fcdns_rjtdetail, tvb, offset+14, 1, + 0); + proto_tree_add_item (req_tree, hf_fcdns_vendor, tvb, offset+15, 1, 0); + } +} + +static void +dissect_fcdns (tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree) +{ + /* Set up structures needed to add the protocol subtree and manage it */ + proto_item *ti = NULL; + proto_tree *fcdns_tree = NULL; + int offset = 0; + int opcode, + failed_opcode = 0; + int isreq = 1; + fc_ct_preamble cthdr; + conversation_t *conversation; + fcdns_conv_data_t *cdata; + fcdns_conv_key_t ckey, *req_key; + + tvb_memcpy (tvb, (guint8 *)&cthdr, offset, FCCT_PRMBL_SIZE); + cthdr.revision = tvb_get_guint8 (tvb, offset); + cthdr.in_id = tvb_get_ntoh24 (tvb, offset+1); + cthdr.opcode = ntohs (cthdr.opcode); + opcode = cthdr.opcode; + cthdr.maxres_size = ntohs (cthdr.maxres_size); + + /* Determine the type of server the request/response is for */ + if (check_col(pinfo->cinfo, COL_PROTOCOL)) { + if (cthdr.gstype == FCCT_GSTYPE_DIRSVC) + col_set_str (pinfo->cinfo, COL_PROTOCOL, "dNS"); + else + col_set_str (pinfo->cinfo, COL_PROTOCOL, "Unzoned NS"); + } + + if (tree) { + if (cthdr.gstype == FCCT_GSTYPE_DIRSVC) { + ti = proto_tree_add_protocol_format (tree, proto_fcdns, tvb, 0, + tvb_reported_length (tvb), + "dNS"); + fcdns_tree = proto_item_add_subtree (ti, ett_fcdns); + } + else { + ti = proto_tree_add_protocol_format (tree, proto_fcdns, tvb, 0, + tvb_reported_length (tvb), + "Unzoned NS"); + fcdns_tree = proto_item_add_subtree (ti, ett_fcdns); + } + } + + if ((opcode != FCCT_MSG_ACC) && (opcode != FCCT_MSG_RJT)) { + conversation = find_conversation (&pinfo->src, &pinfo->dst, + pinfo->ptype, pinfo->oxid, + pinfo->rxid, NO_PORT2); + if (!conversation) { + conversation = conversation_new (&pinfo->src, &pinfo->dst, + pinfo->ptype, pinfo->oxid, + pinfo->rxid, NO_PORT2); + } + + ckey.conv_idx = conversation->index; + + cdata = (fcdns_conv_data_t *)g_hash_table_lookup (fcdns_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 = g_mem_chunk_alloc (fcdns_req_keys); + req_key->conv_idx = conversation->index; + + cdata = g_mem_chunk_alloc (fcdns_req_vals); + cdata->opcode = opcode; + + g_hash_table_insert (fcdns_req_hash, req_key, cdata); + } + if (check_col (pinfo->cinfo, COL_INFO)) { + col_set_str (pinfo->cinfo, COL_INFO, val_to_str (opcode, fc_dns_opcode_val, + "0x%x")); + } + } + else { + /* Opcode is ACC or RJT */ + conversation = find_conversation (&pinfo->src, &pinfo->dst, + pinfo->ptype, pinfo->oxid, + pinfo->rxid, NO_PORT2); + isreq = 0; + if (!conversation) { + if (tree && (opcode == FCCT_MSG_ACC)) { + if (check_col (pinfo->cinfo, COL_INFO)) { + col_set_str (pinfo->cinfo, COL_INFO, + val_to_str (opcode, fc_dns_opcode_val, + "0x%x")); + } + /* No record of what this accept is for. Can't decode */ + proto_tree_add_text (fcdns_tree, tvb, 0, tvb_length (tvb), + "No record of Exchg. Unable to decode MSG_ACC/RJT"); + return; + } + } + else { + ckey.conv_idx = conversation->index; + + cdata = (fcdns_conv_data_t *)g_hash_table_lookup (fcdns_req_hash, &ckey); + + if (cdata != NULL) { + if (opcode == FCCT_MSG_ACC) { + opcode = cdata->opcode; + } + else + failed_opcode = cdata->opcode; + } + + if (check_col (pinfo->cinfo, COL_INFO)) { + if (opcode != FCCT_MSG_RJT) { + col_add_fstr (pinfo->cinfo, COL_INFO, "ACC (%s)", + val_to_str (opcode, fc_dns_opcode_val, + "0x%x")); + } + else { + col_add_fstr (pinfo->cinfo, COL_INFO, "RJT (%s)", + val_to_str (failed_opcode, + fc_dns_opcode_val, + "0x%x")); + } + } + + if (tree) { + if ((cdata == NULL) && (opcode != FCCT_MSG_RJT)) { + /* No record of what this accept is for. Can't decode */ + proto_tree_add_text (fcdns_tree, tvb, 0, tvb_length (tvb), + "No record of Exchg. Unable to decode MSG_ACC/RJT"); + return; + } + } + } + } + + if (tree) { + proto_tree_add_item (fcdns_tree, hf_fcdns_opcode, tvb, offset+8, 2, 0); + proto_tree_add_item (fcdns_tree, hf_fcdns_maxres_size, tvb, offset+10, + 2, 0); + } + + switch (opcode) { + case FCCT_MSG_RJT: + dissect_fcdns_rjt (tvb, fcdns_tree); + break; + case FCDNS_GA_NXT: + dissect_fcdns_ganxt (tvb, fcdns_tree, isreq); + break; + case FCDNS_GPN_ID: + dissect_fcdns_gpnid (tvb, fcdns_tree, isreq); + break; + case FCDNS_GNN_ID: + dissect_fcdns_gnnid (tvb, fcdns_tree, isreq); + break; + case FCDNS_GCS_ID: + dissect_fcdns_gcsid (tvb, fcdns_tree, isreq); + break; + case FCDNS_GFT_ID: + dissect_fcdns_gftid (tvb, fcdns_tree, isreq); + break; + case FCDNS_GSPN_ID: + dissect_fcdns_gspnid (tvb, fcdns_tree, isreq); + break; + case FCDNS_GPT_ID: + dissect_fcdns_gptid (tvb, fcdns_tree, isreq); + break; + case FCDNS_GFPN_ID: + dissect_fcdns_gfpnid (tvb, fcdns_tree, isreq); + break; + case FCDNS_GFD_ID: + dissect_fcdns_gfdid (tvb, fcdns_tree, isreq); + break; + case FCDNS_GFF_ID: + dissect_fcdns_gffid (tvb, fcdns_tree, isreq); + break; + case FCDNS_GID_PN: + dissect_fcdns_gidpn (tvb, fcdns_tree, isreq); + break; + case FCDNS_GIPP_PN: + dissect_fcdns_gipppn (tvb, fcdns_tree, isreq); + break; + case FCDNS_GID_NN: + dissect_fcdns_gidnn (tvb, fcdns_tree, isreq); + break; + case FCDNS_GPN_NN: + dissect_fcdns_gpnnn (tvb, fcdns_tree, isreq); + break; + case FCDNS_GIP_NN: + dissect_fcdns_gipnn (tvb, fcdns_tree, isreq); + break; + case FCDNS_GSNN_NN: + dissect_fcdns_gsnnnn (tvb, fcdns_tree, isreq); + break; + case FCDNS_GID_FT: + dissect_fcdns_gidft (tvb, fcdns_tree, isreq); + break; + case FCDNS_GPN_FT: + dissect_fcdns_gpnft (tvb, fcdns_tree, isreq); + break; + case FCDNS_GNN_FT: + dissect_fcdns_gnnft (tvb, fcdns_tree, isreq); + break; + case FCDNS_GID_PT: + dissect_fcdns_gidpt (tvb, fcdns_tree, isreq); + break; + case FCDNS_GID_IPP: + dissect_fcdns_gidipp (tvb, fcdns_tree, isreq); + break; + case FCDNS_GID_FF: + dissect_fcdns_gidff (tvb, fcdns_tree, isreq); + break; + case FCDNS_RPN_ID: + dissect_fcdns_rpnid (tvb, fcdns_tree, isreq); + break; + case FCDNS_RNN_ID: + dissect_fcdns_rnnid (tvb, fcdns_tree, isreq); + break; + case FCDNS_RCS_ID: + dissect_fcdns_rcsid (tvb, fcdns_tree, isreq); + break; + case FCDNS_RPT_ID: + dissect_fcdns_rptid (tvb, fcdns_tree, isreq); + break; + case FCDNS_RFT_ID: + dissect_fcdns_rftid (tvb, fcdns_tree, isreq); + break; + case FCDNS_RSPN_ID: + dissect_fcdns_rspnid (tvb, fcdns_tree, isreq); + break; + case FCDNS_RIPP_ID: + dissect_fcdns_rippid (tvb, fcdns_tree, isreq); + break; + case FCDNS_RFD_ID: + dissect_fcdns_rfdid (tvb, fcdns_tree, isreq); + break; + case FCDNS_RFF_ID: + dissect_fcdns_rffid (tvb, fcdns_tree, isreq); + break; + case FCDNS_RIP_NN: + dissect_fcdns_ripnn (tvb, fcdns_tree, isreq); + break; + case FCDNS_RSNN_NN: + dissect_fcdns_rsnnnn (tvb, fcdns_tree, isreq); + break; + case FCDNS_DA_ID: + dissect_fcdns_daid (tvb, fcdns_tree, isreq); + break; + case FCDNS_GE_ID: + dissect_fcdns_geid (tvb, fcdns_tree, isreq); + break; + case FCDNS_GE_PN: + dissect_fcdns_gepn (tvb, fcdns_tree, isreq); + break; + case FCDNS_GE_NN: + dissect_fcdns_genn (tvb, fcdns_tree, isreq); + break; + case FCDNS_GE_IP: + dissect_fcdns_geip (tvb, fcdns_tree, isreq); + break; + case FCDNS_GE_FT: + dissect_fcdns_geft (tvb, fcdns_tree, isreq); + break; + case FCDNS_GE_PT: + dissect_fcdns_gept (tvb, fcdns_tree, isreq); + break; + case FCDNS_GE_ZM: + dissect_fcdns_gezm (tvb, fcdns_tree, isreq); + break; + case FCDNS_GE_ZN: + dissect_fcdns_gezn (tvb, fcdns_tree, isreq); + break; + case FCDNS_GE_IPP: + dissect_fcdns_geipp (tvb, fcdns_tree, isreq); + break; + case FCDNS_GE_FF: + dissect_fcdns_geff (tvb, fcdns_tree, isreq); + break; + default: + break; + } +} + +/* Register the protocol with Ethereal */ + +/* this format is required because a script is used to build the C function + that calls all the protocol registration. +*/ + +void +proto_register_fcdns (void) +{ + +/* Setup list of header fields See Section 1.6.1 for details*/ + static hf_register_info hf[] = { + { &hf_fcdns_gssubtype, + {"GS_Subtype", "fcdns.gssubtype", FT_UINT8, BASE_HEX, + VALS (fc_dns_subtype_val), 0x0, "", HFILL}}, + {&hf_fcdns_opcode, + {"Opcode", "fcdns.opcode", FT_UINT16, BASE_HEX, VALS (fc_dns_opcode_val), + 0x0, "", HFILL}}, + { &hf_fcdns_req_portid, + {"Port Identifier", "fcdns.req.portid", FT_STRING, BASE_HEX, NULL, 0x0, + "", HFILL}}, + { &hf_fcdns_rply_pname, + {"Port Name", "fcdns.rply.pname", FT_STRING, BASE_HEX, NULL, 0x0, "", + HFILL}}, + { &hf_fcdns_rply_nname, + {"Node Name", "fcdns.rply.nname", FT_STRING, BASE_HEX, NULL, 0x0, "", + HFILL}}, + { &hf_fcdns_rply_cos, + {"Class of Service Supported", "fcdns.rply.cos", FT_STRING, BASE_HEX, + NULL, 0x0, "", HFILL}}, + { &hf_fcdns_rply_gft, + {"FC-4 Types Supported", "fcdns.rply.fc4type", FT_STRING, BASE_HEX, + NULL, 0x0, "", HFILL}}, + { &hf_fcdns_rply_snamelen, + {"Symbolic Node Name Length", "fcdns.rply.snamelen", FT_UINT8, BASE_DEC, + NULL, 0x0, "", HFILL}}, + { &hf_fcdns_rply_sname, + {"Symbolic Node Name", "fcdns.rply.sname", FT_STRING, BASE_HEX, NULL, + 0x0, "", HFILL}}, + { &hf_fcdns_rply_ptype, + {"Port Type", "fcdns.rply.porttype", FT_UINT8, BASE_HEX, + VALS (fc_dns_port_type_val), 0x0, "", HFILL}}, + { &hf_fcdns_rply_fpname, + {"Fabric Port Name", "fcdns.rply.fpname", FT_STRING, BASE_HEX, NULL, + 0x0, "", HFILL}}, + { &hf_fcdns_fc4type, + {"FC-4 Type", "fcdns.req.fc4type", FT_STRING, BASE_HEX, NULL, 0x0, + "", HFILL}}, + { &hf_fcdns_rply_fc4feat, + {"FC-4 Features", "fcdns.rply.fc4features", FT_STRING, BASE_HEX, + NULL, 0x0, "", HFILL}}, + { &hf_fcdns_req_pname, + {"Port Name", "fcdns.req.portname", FT_STRING, BASE_HEX, NULL, 0x0, + "", HFILL}}, + { &hf_fcdns_rply_portid, + {"Port Identifier", "fcdns.rply.portid", FT_STRING, BASE_HEX, NULL, + 0x0, "", HFILL}}, + { &hf_fcdns_req_nname, + {"Node Name", "fcdns.req.nname", FT_STRING, BASE_HEX, NULL, 0x0, + "", HFILL}}, + { &hf_fcdns_req_domainscope, + {"Domain ID Scope", "fcdns.req.domainid", FT_UINT8, BASE_HEX, NULL, + 0x0, "", HFILL}}, + { &hf_fcdns_req_areascope, + {"Area ID Scope", "fcdns.req.areaid", FT_UINT8, BASE_HEX, NULL, + 0x0, "", HFILL}}, + { &hf_fcdns_req_fc4type, + {"FC-4 Type", "fcdns.req.fc4type", FT_UINT8, BASE_HEX, + VALS (fc_fc4_val), 0x0, "", HFILL}}, + { &hf_fcdns_req_ptype, + {"Port Type", "fcdns.req.porttype", FT_UINT8, BASE_HEX, + VALS (fc_dns_port_type_val), 0x0, "", HFILL}}, + { &hf_fcdns_req_ip, + {"IP Address", "fcdns.req.ip", FT_IPv6, BASE_DEC, NULL, 0x0, + "", HFILL}}, + { &hf_fcdns_req_fc4feature, + {"FC-4 Feature Bits", "fcdns.req.fc4feature", FT_STRING, + BASE_HEX, NULL, 0x0, "", HFILL}}, + { &hf_fcdns_req_cos, + {"Class of Service Supported", "fcdns.req.class", FT_STRING, + BASE_HEX, NULL, 0x0, "", HFILL}}, + { &hf_fcdns_req_fc4types, + {"FC-4 TYPEs Supported", "fcdns.req.fc4types", FT_STRING, + BASE_HEX, NULL, 0x0, "", HFILL}}, + { &hf_fcdns_rply_fc4type, + {"FC-4 Descriptor Type", "fcdns.rply.fc4type", FT_UINT8, BASE_HEX, + VALS (fc_fc4_val), 0x0, "", HFILL}}, + { &hf_fcdns_req_snamelen, + {"Symbolic Name Length", "fcdns.req.snamelen", FT_UINT8, BASE_DEC, + NULL, 0x0, "", HFILL}}, + { &hf_fcdns_req_sname, + {"Symbolic Port Name", "fcdns.req.sname", FT_STRING, BASE_HEX, NULL, + 0x0, "", HFILL}}, + { &hf_fcdns_rply_spnamelen, + {"Symbolic Port Name Length", "fcdns.rply.spnamelen", FT_UINT8, + BASE_DEC, NULL, 0x0, "", HFILL}}, + {&hf_fcdns_rply_spname, + {"Symbolic Port Name", "fcdns.rply.spname", FT_STRING, BASE_HEX, NULL, + 0x0, "", HFILL}}, + { &hf_fcdns_rply_ipa, + {"Initial Process Associator", "fcdns.rply.ipa", FT_BYTES, BASE_HEX, + NULL, 0x0, "", HFILL}}, + { &hf_fcdns_rply_ipnode, + {"Node IP Address", "fcdns.rply.ipnode", FT_IPv6, BASE_DEC, NULL, + 0x0, "", HFILL}}, + { &hf_fcdns_rply_ipport, + {"Port IP Address", "fcdns.rply.ipport", FT_IPv6, BASE_DEC, NULL, + 0x0, "", HFILL}}, + { &hf_fcdns_rply_fc4desclen, + {"FC-4 Descriptor Length", "fcdns.rply.fc4desclen", FT_UINT8, + BASE_DEC, NULL, 0x0, "", HFILL}}, + { &hf_fcdns_rply_fc4desc, + {"FC-4 Descriptor", "fcdns.rply.fc4desc", FT_BYTES, BASE_HEX, NULL, + 0x0, "", HFILL}}, + { &hf_fcdns_rply_hrdaddr, + {"Hard Address", "fcdns.rply.hrdaddr", FT_STRING, BASE_HEX, NULL, + 0x0, "", HFILL}}, + { &hf_fcdns_req_fdesclen, + {"FC-4 Descriptor Length", "fcdns.req.fc4desclen", FT_UINT8, BASE_DEC, + NULL, 0x0, "", HFILL}}, + { &hf_fcdns_req_fdesc, + {"FC-4 Descriptor", "fcdns.req.fc4desc", FT_STRING, BASE_HEX, NULL, + 0x0, "", HFILL}}, + { &hf_fcdns_req_spnamelen, + {"Symbolic Port Name Length", "fcdns.req.spnamelen", FT_UINT8, + BASE_DEC, NULL, 0x0, "", HFILL}}, + { &hf_fcdns_req_spname, + {"Symbolic Port Name", "fcdns.req.spname", FT_STRING, BASE_HEX, NULL, + 0x0, "", HFILL}}, + { &hf_fcdns_reason, + {"Reason Code", "fcdns.rply.reason", FT_UINT8, BASE_HEX, + VALS (fc_ct_rjt_code_vals), 0x0, "", HFILL}}, + { &hf_fcdns_rjtdetail, + {"Reason Code Explanantion", "fcdns.rply.reasondet", FT_UINT8, + BASE_HEX, VALS (fc_dns_rjt_det_code_val), 0x0, "", HFILL}}, + { &hf_fcdns_vendor, + {"Vendor Unique Reject Code", "fcdns.rply.vendor", FT_UINT8, + BASE_HEX, NULL, 0x0, "", HFILL}}, + { &hf_fcdns_zone_mbrtype, + {"Zone Member Type", "fcdns.zone.mbrtype", FT_UINT8, BASE_HEX, + VALS (fc_swils_zonembr_type_val), 0x0, "", HFILL}}, + { &hf_fcdns_zone_mbrid, + {"Member Identifier", "swils.zone.mbrid", FT_STRING, BASE_HEX, NULL, + 0x0, "", HFILL}}, + { &hf_fcdns_zonenm, + {"Zone Name", "fcdns.zonename", FT_STRING, BASE_HEX, NULL, 0x0, "", + HFILL}}, + { &hf_fcdns_portip, + {"Port IP Address", "fcdns.portip", FT_IPv4, BASE_DEC, NULL, 0x0, + "", HFILL}}, + { &hf_fcdns_sw2_objfmt, + {"Name Entry Object Format", "fcdns.entry.objfmt", FT_UINT8, BASE_HEX, + NULL, 0x0, "", HFILL}}, + { &hf_fcdns_num_fc4desc, + {"Number of FC4 Descriptors Registered", "fcdns.entry.numfc4desc", + FT_UINT8, BASE_DEC, NULL, 0x0, "", HFILL}}, + { &hf_fcdns_rply_ownerid, + {"Owner Id", "fcdns.rply.ownerid", FT_STRING, BASE_HEX, NULL, 0x0, "", + HFILL}}, + { &hf_fcdns_maxres_size, + {"Maximum/Residual Size", "fcdns.maxres_size", FT_UINT16, BASE_DEC, + NULL, 0x0, "", HFILL}}, + }; + + static gint *ett[] = { + &ett_fcdns, + }; + + /* Register the protocol name and description */ + proto_fcdns = proto_register_protocol("Fibre Channel Name Server", + "FC-dNS", "FC-dNS"); + + /* Required function calls to register the header fields and subtrees used */ + proto_register_field_array(proto_fcdns, hf, array_length(hf)); + proto_register_subtree_array(ett, array_length(ett)); + register_init_routine (&fcdns_init_protocol); +} + +/* If this dissector uses sub-dissector registration add a registration routine. + This format is required because a script is used to find these routines and + create the code that calls these routines. +*/ +void +proto_reg_handoff_fcdns (void) +{ + dissector_handle_t dns_handle; + + dns_handle = create_dissector_handle (dissect_fcdns, proto_fcdns); + dissector_add("fcct.server", FCCT_GSRVR_DNS, dns_handle); + dissector_add("fcct.server", FCCT_GSRVR_UNS, dns_handle); + + data_handle = find_dissector ("data"); +} + + |