wireshark/epan/dissectors/packet-ziop.c
Michael Mann 1e60d63c8c Create call_data_dissector() to call data dissector.
This saves many dissectors the need to find the data dissector and store a handle to it.

There were also some that were finding it, but not using it.
For others this was the only reason for their handoff function, so it could be eliminated.

Change-Id: I5d3f951ee1daa3d30c060d21bd12bbc881a8027b
Reviewed-on: https://code.wireshark.org/review/14530
Petri-Dish: Michael Mann <mmann78@netscape.net>
Reviewed-by: Michael Mann <mmann78@netscape.net>
2016-03-20 17:38:03 +00:00

349 lines
10 KiB
C

/* packet-ziop.c
* Routines for CORBA ZIOP packet disassembly
* Significantly based on packet-giop.c
* Copyright 2009 Alvaro Vega Garcia <avega at tid dot es>
*
* According with GIOP Compression RFP revised submission
* OMG mars/2008-12-20
* http://www.omg.org/docs/ptc/09-01-03.pdf
*
* Wireshark - Network traffic analyzer
* By Gerald Combs <gerald@wireshark.org>
* Copyright 1998 Gerald Combs
*
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#include "config.h"
#include <epan/packet.h>
#include <epan/expert.h>
#include "packet-ziop.h"
#include "packet-giop.h"
#include "packet-tcp.h"
/*
* Set to 1 for DEBUG output - TODO make this a runtime option
*/
#define DEBUG 0
void proto_reg_handoff_ziop(void);
void proto_register_ziop(void);
/*
* ------------------------------------------------------------------------------------------+
* Data/Variables/Structs
* ------------------------------------------------------------------------------------------+
*/
static int proto_ziop = -1;
/*
* (sub)Tree declares
*/
static gint hf_ziop_magic = -1;
static gint hf_ziop_giop_version_major = -1;
static gint hf_ziop_giop_version_minor = -1;
static gint hf_ziop_flags = -1;
static gint hf_ziop_message_type = -1;
static gint hf_ziop_message_size = -1;
static gint hf_ziop_compressor_id = -1;
static gint hf_ziop_original_length = -1;
static gint ett_ziop = -1;
static expert_field ei_ziop_version = EI_INIT;
static dissector_handle_t ziop_tcp_handle;
static const value_string ziop_compressor_ids[] = {
{ 0, "None" },
{ 1, "GZIP"},
{ 2, "PKZIP"},
{ 3, "BZIP2"},
{ 4, "ZLIB"},
{ 5, "LZMA"},
{ 6, "LZOP"},
{ 7, "RZIP"},
{ 8, "7X"},
{ 9, "XAR"},
{ 0, NULL}
};
static const value_string giop_message_types[] = {
{ 0x0, "Request" },
{ 0x1, "Reply"},
{ 0x2, "CancelRequest"},
{ 0x3, "LocateRequest"},
{ 0x4, "LocateReply"},
{ 0x5, "CloseConnection"},
{ 0x6, "MessageError"},
{ 0x7, "Fragment"},
{ 0, NULL}
};
static gboolean ziop_desegment = TRUE;
/* Main entry point */
static int
dissect_ziop (tvbuff_t * tvb, packet_info * pinfo, proto_tree * tree, void* data _U_) {
guint offset = 0;
guint8 giop_version_major, giop_version_minor, message_type;
proto_tree *ziop_tree = NULL;
proto_item *ti;
guint8 flags;
guint byte_order;
const char *label = "none";
if (tvb_reported_length(tvb) < 7)
return 0;
col_set_str (pinfo->cinfo, COL_PROTOCOL, ZIOP_MAGIC);
/* Clear out stuff in the info column */
col_clear(pinfo->cinfo, COL_INFO);
ti = proto_tree_add_item (tree, proto_ziop, tvb, 0, -1, ENC_NA);
ziop_tree = proto_item_add_subtree (ti, ett_ziop);
proto_tree_add_item(ziop_tree, hf_ziop_magic, tvb, offset, 4, ENC_ASCII|ENC_NA);
offset += 4;
proto_tree_add_item(ziop_tree, hf_ziop_giop_version_major, tvb, offset, 1, ENC_BIG_ENDIAN);
giop_version_major = tvb_get_guint8(tvb, offset);
offset++;
proto_tree_add_item(ziop_tree, hf_ziop_giop_version_minor, tvb, offset, 1, ENC_BIG_ENDIAN);
giop_version_minor = tvb_get_guint8(tvb, offset);
offset++;
if ( (giop_version_major < 1) ||
(giop_version_minor < 2) ) /* earlier than GIOP 1.2 */
{
col_add_fstr (pinfo->cinfo, COL_INFO, "Version %u.%u",
giop_version_major, giop_version_minor);
expert_add_info_format(pinfo, ti, &ei_ziop_version,
"Version %u.%u not supported",
giop_version_major,
giop_version_minor);
call_data_dissector(tvb, pinfo, tree);
return tvb_reported_length(tvb);
}
flags = tvb_get_guint8(tvb, offset);
byte_order = (flags & 0x01) ? ENC_LITTLE_ENDIAN : ENC_BIG_ENDIAN;
if (flags & 0x01) {
label = "little-endian";
}
proto_tree_add_uint_format_value(ziop_tree, hf_ziop_flags, tvb, offset, 1,
flags, "0x%02x (%s)", flags, label);
offset++;
proto_tree_add_item(ziop_tree, hf_ziop_message_type, tvb, offset, 1, ENC_BIG_ENDIAN);
message_type = tvb_get_guint8(tvb, offset);
offset++;
col_add_fstr (pinfo->cinfo, COL_INFO, "ZIOP %u.%u %s",
giop_version_major,
giop_version_minor,
val_to_str(message_type, giop_message_types,
"Unknown message type (0x%02x)")
);
proto_tree_add_item(ziop_tree, hf_ziop_message_size, tvb, offset, 4, byte_order);
offset += 4;
proto_tree_add_item(ziop_tree, hf_ziop_compressor_id, tvb, offset, 2, byte_order);
offset += 4;
proto_tree_add_item(ziop_tree, hf_ziop_original_length, tvb, offset, 4, byte_order);
return tvb_reported_length(tvb);
}
static guint
get_ziop_pdu_len(packet_info *pinfo _U_, tvbuff_t *tvb, int offset, void *data _U_)
{
guint8 flags;
guint message_size;
gboolean stream_is_big_endian;
if ( tvb_memeql(tvb, 0, ZIOP_MAGIC, 4) != 0)
return 0;
flags = tvb_get_guint8(tvb, offset + 6);
stream_is_big_endian = ((flags & 0x1) == 0);
if (stream_is_big_endian)
message_size = tvb_get_ntohl(tvb, offset + 8);
else
message_size = tvb_get_letohl(tvb, offset + 8);
return message_size + ZIOP_HEADER_SIZE;
}
static int
dissect_ziop_tcp (tvbuff_t * tvb, packet_info * pinfo, proto_tree * tree, void* data)
{
if ( tvb_memeql(tvb, 0, ZIOP_MAGIC, 4) != 0)
{
if (tvb_get_ntohl(tvb, 0) == GIOP_MAGIC_NUMBER)
{
dissect_giop(tvb, pinfo, tree);
return tvb_captured_length(tvb);
}
return 0;
}
tcp_dissect_pdus(tvb, pinfo, tree, ziop_desegment, ZIOP_HEADER_SIZE,
get_ziop_pdu_len, dissect_ziop, data);
return tvb_captured_length(tvb);
}
gboolean
dissect_ziop_heur (tvbuff_t * tvb, packet_info * pinfo, proto_tree * tree, void * data)
{
guint tot_len;
conversation_t *conversation;
/* check magic number and version */
tot_len = tvb_captured_length(tvb);
if (tot_len < ZIOP_HEADER_SIZE) /* tot_len < 12 */
{
/* Not enough data captured to hold the ZIOP header; don't try
to interpret it as GIOP. */
return FALSE;
}
if ( tvb_memeql(tvb, 0, ZIOP_MAGIC, 4) != 0)
{
return FALSE;
}
if ( pinfo->ptype == PT_TCP )
{
/*
* Make the ZIOP dissector the dissector for this conversation.
*
* If this isn't the first time this packet has been processed,
* we've already done this work, so we don't need to do it
* again.
*/
if (!pinfo->fd->flags.visited)
{
conversation = find_or_create_conversation(pinfo);
/* Set dissector */
conversation_set_dissector(conversation, ziop_tcp_handle);
}
dissect_ziop_tcp (tvb, pinfo, tree, data);
}
else
{
dissect_ziop (tvb, pinfo, tree, data);
}
return TRUE;
}
void
proto_register_ziop (void)
{
/* A header field is something you can search/filter on.
*
* We create a structure to register our fields. It consists of an
* array of hf_register_info structures, each of which are of the format
* {&(field id), {name, abbrev, type, display, strings, bitmask, blurb, HFILL}}.
*/
static hf_register_info hf[] = {
{ &hf_ziop_magic,
{ "Header magic", "ziop.magic", FT_STRING, BASE_NONE, NULL, 0x0,
"ZIOPHeader magic", HFILL }},
{ &hf_ziop_giop_version_major,
{ "Header major version", "ziop.giop_version_major", FT_UINT8, BASE_OCT, NULL, 0x0,
"ZIOPHeader giop_major_version", HFILL }},
{ &hf_ziop_giop_version_minor,
{ "Header minor version", "ziop.giop_version_minor", FT_UINT8, BASE_OCT, NULL, 0x0,
"ZIOPHeader giop_minor_version", HFILL }},
{ &hf_ziop_flags,
{ "Header flags", "ziop.flags", FT_UINT8, BASE_OCT, NULL, 0x0,
"ZIOPHeader flags", HFILL }},
{ &hf_ziop_message_type,
{ "Header type", "ziop.message_type", FT_UINT8, BASE_OCT, VALS(giop_message_types), 0x0,
"ZIOPHeader message_type", HFILL }},
{ &hf_ziop_message_size,
{ "Header size", "ziop.message_size", FT_UINT32, BASE_DEC, NULL, 0x0,
"ZIOPHeader message_size", HFILL }},
{ &hf_ziop_compressor_id,
{ "Header compressor id", "ziop.compressor_id", FT_UINT16, BASE_DEC, VALS(ziop_compressor_ids), 0x0,
"ZIOPHeader compressor_id", HFILL }},
{ &hf_ziop_original_length,
{ "Header original length", "ziop.original_length", FT_UINT32, BASE_DEC, NULL, 0x0,
"ZIOP original_length", HFILL }}
};
static gint *ett[] = {
&ett_ziop
};
static ei_register_info ei[] = {
{ &ei_ziop_version, { "ziop.version_not_supported", PI_PROTOCOL, PI_WARN, "Version not supported", EXPFILL }},
};
expert_module_t* expert_ziop;
proto_ziop = proto_register_protocol("Zipped Inter-ORB Protocol", "ZIOP",
"ziop");
proto_register_field_array (proto_ziop, hf, array_length (hf));
proto_register_subtree_array (ett, array_length (ett));
expert_ziop = expert_register_protocol(proto_ziop);
expert_register_field_array(expert_ziop, ei, array_length(ei));
register_dissector("ziop", dissect_ziop, proto_ziop);
}
void
proto_reg_handoff_ziop (void)
{
ziop_tcp_handle = create_dissector_handle(dissect_ziop_tcp, proto_ziop);
dissector_add_for_decode_as("udp.port", ziop_tcp_handle);
heur_dissector_add("tcp", dissect_ziop_heur, "ZIOP over TCP", "ziop_tcp", proto_ziop, HEURISTIC_ENABLE);
}
/*
* Editor modelines - http://www.wireshark.org/tools/modelines.html
*
* Local variables:
* c-basic-offset: 2
* tab-width: 8
* indent-tabs-mode: nil
* End:
*
* vi: set shiftwidth=2 tabstop=8 expandtab:
* :indentSize=2:tabSize=8:noTabs=true:
*/