aboutsummaryrefslogtreecommitdiffstats
path: root/plugins/docsis/packet-tlv-cmctrl.c
diff options
context:
space:
mode:
authorJaap Keuter <jaap.keuter@xs4all.nl>2010-12-16 07:01:22 +0000
committerJaap Keuter <jaap.keuter@xs4all.nl>2010-12-16 07:01:22 +0000
commita6b55e6530d7af3afba0af7480cb0e7c455a463b (patch)
tree31d41299e17e1f60c598523e122699d805121dec /plugins/docsis/packet-tlv-cmctrl.c
parent5ae824fc8fdd1c2741fb10c1f03e4fd9dbba010f (diff)
From Guido Reismueller:
packet-mdd.c - TLV 15 fixed packet-tlv.c - Added Support for DOCSIS 3.0 TLVs (38-66) Added Support for the following DOCSIS 3.0 MAC Management Messages - DBC-REQ (packet-dbcreq.c) - DBC-RSP (packet-dbcrsp.c) - DBC-ACK (packet-dbcack.c) - CM-CTRL-REQ (packet-cmctrlreq.c) - CM-CTRL-RSP (packet-cmctrlrsp.c) - DPV-REQ (packet-dpvreq.c) - DPV-RSP (packet-dpvrsp.c) svn path=/trunk/; revision=35193
Diffstat (limited to 'plugins/docsis/packet-tlv-cmctrl.c')
-rw-r--r--plugins/docsis/packet-tlv-cmctrl.c355
1 files changed, 355 insertions, 0 deletions
diff --git a/plugins/docsis/packet-tlv-cmctrl.c b/plugins/docsis/packet-tlv-cmctrl.c
new file mode 100644
index 0000000000..b466ac465e
--- /dev/null
+++ b/plugins/docsis/packet-tlv-cmctrl.c
@@ -0,0 +1,355 @@
+/* packet-tlv-cmctrl.c
+ * Routines to Dissect TLV's for CM-Control Messages
+ * Copyright 2010, Guido Reismueller <g.reismueller[AT]avm.de>
+ *
+ * $Id$
+ *
+ * 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+ */
+
+#ifdef HAVE_CONFIG_H
+# include "config.h"
+#endif
+
+#include <epan/packet.h>
+
+#define CM_CTRL_MUTE 1
+#define CM_CTRL_MUTE_TIMEOUT 2
+#define CM_CTRL_REINIT 3
+#define CM_CTRL_DISABLE_FWD 4
+#define CM_CTRL_DS_EVENT 5
+#define CM_CTRL_US_EVENT 6
+#define CM_CTRL_EVENT 7
+
+#define DS_EVENT_CH_ID 1
+#define DS_EVENT_MASK 2
+
+#define US_EVENT_CH_ID 1
+#define US_EVENT_MASK 2
+
+static int proto_cmctrl_tlv = -1;
+static int hf_cmctrl_tlv_mute = -1;
+static int hf_cmctrl_tlv_mute_timeout = -1;
+static int hf_cmctrl_tlv_reinit = -1;
+static int hf_cmctrl_tlv_disable_fwd = -1;
+static int hf_cmctrl_tlv_ds_event = -1;
+static int hf_cmctrl_tlv_us_event = -1;
+static int hf_cmctrl_tlv_event = -1;
+
+static int hf_ds_event_ch_id = -1;
+static int hf_ds_event_mask = -1;
+
+static int hf_us_event_ch_id = -1;
+static int hf_us_event_mask = -1;
+
+static gint ett_cmctrl_tlv = -1;
+static gint ett_cmctrl_tlv_ds_event = -1;
+static gint ett_cmctrl_tlv_us_event = -1;
+
+
+static void
+dissect_ds_event(tvbuff_t * tvb, proto_tree *tree, int start, guint16 len)
+{
+ guint8 type, length;
+ proto_item *it;
+ proto_tree *event_tree;
+ int pos = start;
+ it =
+ proto_tree_add_text (tree, tvb, start, len,
+ "Override Downstream Status Event Event Mask (Length = %u)", len);
+ event_tree = proto_item_add_subtree (it, ett_cmctrl_tlv_ds_event);
+
+ while (pos < (start + len))
+ {
+ type = tvb_get_guint8 (tvb, pos++);
+ length = tvb_get_guint8 (tvb, pos++);
+ switch (type)
+ {
+ case DS_EVENT_CH_ID:
+ if (length == 1)
+ {
+ proto_tree_add_item (event_tree, hf_ds_event_ch_id,
+ tvb, pos, length, FALSE);
+ }
+ else
+ {
+ THROW (ReportedBoundsError);
+ }
+ break;
+ case DS_EVENT_MASK:
+ if (length == 2)
+ {
+ proto_tree_add_item (event_tree, hf_ds_event_mask,
+ tvb, pos, length, FALSE);
+ }
+ else
+ {
+ THROW (ReportedBoundsError);
+ }
+ break;
+ } /* switch */
+ pos = pos + length;
+ } /* while */
+}
+
+static void
+dissect_us_event(tvbuff_t * tvb, proto_tree *tree, int start, guint16 len)
+{
+ guint8 type, length;
+ proto_item *it;
+ proto_tree *event_tree;
+ int pos = start;
+ it =
+ proto_tree_add_text (tree, tvb, start, len,
+ "Override Upstream Status Enable Event Mask (Length = %u)", len);
+ event_tree = proto_item_add_subtree (it, ett_cmctrl_tlv_us_event);
+
+ while (pos < (start + len))
+ {
+ type = tvb_get_guint8 (tvb, pos++);
+ length = tvb_get_guint8 (tvb, pos++);
+ switch (type)
+ {
+ case US_EVENT_CH_ID:
+ if (length == 1)
+ {
+ proto_tree_add_item (event_tree, hf_us_event_ch_id,
+ tvb, pos, length, FALSE);
+ }
+ else
+ {
+ THROW (ReportedBoundsError);
+ }
+ break;
+ case US_EVENT_MASK:
+ if (length == 2)
+ {
+ proto_tree_add_item (event_tree, hf_us_event_mask,
+ tvb, pos, length, FALSE);
+ }
+ else
+ {
+ THROW (ReportedBoundsError);
+ }
+ break;
+ } /* switch */
+ pos = pos + length;
+ } /* while */
+}
+
+static void
+dissect_cmctrl_tlv (tvbuff_t * tvb, packet_info * pinfo _U_, proto_tree * tree)
+{
+
+ proto_item *it;
+ proto_tree *tlv_tree;
+ int pos = 0;
+ gint total_len;
+ guint8 type, length;
+
+ total_len = tvb_reported_length_remaining (tvb, 0);
+
+ it =
+ proto_tree_add_protocol_format (tree, proto_cmctrl_tlv, tvb, 0,
+ total_len, "TLV Data");
+ tlv_tree = proto_item_add_subtree (it, ett_cmctrl_tlv);
+
+ while (pos < total_len)
+ {
+ type = tvb_get_guint8 (tvb, pos++);
+ length = tvb_get_guint8 (tvb, pos++);
+ switch (type)
+ {
+ case CM_CTRL_MUTE:
+ if (length == 1)
+ {
+ proto_tree_add_item (tlv_tree, hf_cmctrl_tlv_mute,
+ tvb, pos, length, FALSE);
+ }
+ else
+ {
+ THROW (ReportedBoundsError);
+ }
+ break;
+ case CM_CTRL_MUTE_TIMEOUT:
+ if (length == 4 || length == 1) /* response TLV always with len 1 */
+ {
+ proto_tree_add_item (tlv_tree, hf_cmctrl_tlv_mute_timeout,
+ tvb, pos, length, FALSE);
+ }
+ else
+ {
+ THROW (ReportedBoundsError);
+ }
+ break;
+ case CM_CTRL_REINIT:
+ if (length == 1)
+ {
+ proto_tree_add_item (tlv_tree, hf_cmctrl_tlv_reinit,
+ tvb, pos, length, FALSE);
+ }
+ else
+ {
+ THROW (ReportedBoundsError);
+ }
+ break;
+ case CM_CTRL_DISABLE_FWD:
+ if (length == 1)
+ {
+ proto_tree_add_item (tlv_tree, hf_cmctrl_tlv_disable_fwd,
+ tvb, pos, length, FALSE);
+ }
+ else
+ {
+ THROW (ReportedBoundsError);
+ }
+ break;
+ case CM_CTRL_DS_EVENT:
+ if (length == 1)
+ proto_tree_add_item (tlv_tree, hf_cmctrl_tlv_ds_event,
+ tvb, pos, length, FALSE);
+ else
+ dissect_ds_event(tvb, tlv_tree, pos, length);
+ break;
+ case CM_CTRL_US_EVENT:
+ if (length == 1)
+ proto_tree_add_item (tlv_tree, hf_cmctrl_tlv_ds_event,
+ tvb, pos, length, FALSE);
+ else
+ dissect_us_event(tvb, tlv_tree, pos, length);
+ break;
+ case CM_CTRL_EVENT:
+ if (length == 2 || length == 1) /* response TLV always with len 1 */
+ {
+ proto_tree_add_item (tlv_tree, hf_cmctrl_tlv_event,
+ tvb, pos, length, FALSE);
+ }
+ else
+ {
+ THROW (ReportedBoundsError);
+ }
+ break;
+
+ } /* switch */
+ pos = pos + length;
+ } /* while */
+}
+
+/* Register the protocol with Wireshark */
+
+/* this format is require because a script is used to build the C function
+ that calls all the protocol registration.
+*/
+
+
+void
+proto_register_cmctrl_tlv (void)
+{
+
+/* Setup list of header fields See Section 1.6.1 for details*/
+ static hf_register_info hf[] = {
+ {&hf_cmctrl_tlv_mute,
+ {"1 Upstream Channel RF Mute", "cmctrl_tlv.mute",
+ FT_UINT8, BASE_DEC, NULL, 0x0,
+ "Upstream Channel RF Mute", HFILL}
+ },
+ {&hf_cmctrl_tlv_mute_timeout,
+ {"2 RF Mute Timeout Interval", "cmctrl_tlv.mute_timeout",
+ FT_UINT32, BASE_DEC, NULL, 0x0,
+ "RF Mute Timeout Interval", HFILL}
+ },
+ {&hf_cmctrl_tlv_reinit,
+ {"3 CM Reinitialize", "cmctrl_tlv.reinit",
+ FT_UINT8, BASE_DEC, NULL, 0x0,
+ "CM Reinitialize", HFILL}
+ },
+ {&hf_cmctrl_tlv_disable_fwd,
+ {"4 Disable Forwarding", "cmctrl_tlv.disable_fwd",
+ FT_UINT8, BASE_DEC, NULL, 0x0,
+ "Disable Forwarding", HFILL}
+ },
+ {&hf_cmctrl_tlv_ds_event,
+ {"5 Override Downstream Events", "cmctrl_tlv.ds_event",
+ FT_BYTES, BASE_NONE, NULL, 0x0,
+ "Override Downstream Events", HFILL}
+ },
+ {&hf_ds_event_ch_id,
+ {".1 Downstream Channel ID", "cmctrl_tlv.ds_event.chid",
+ FT_UINT8, BASE_DEC, NULL, 0x0,
+ "Downstream Channel ID", HFILL}
+ },
+ {&hf_ds_event_mask,
+ {".2 Downstream Status Event Enable Bitmask", "cmctrl_tlv.ds_event.mask",
+ FT_BYTES, BASE_NONE, NULL, 0x0,
+ "Downstream Status Event Enable Bitmask", HFILL}
+ },
+ {&hf_cmctrl_tlv_us_event,
+ {"6 Override Upstream Events", "cmctrl_tlv.us_event",
+ FT_BYTES, BASE_NONE, NULL, 0x0,
+ "Override Downstream Events", HFILL}
+ },
+ {&hf_us_event_ch_id,
+ {".1 Upstream Channel ID", "cmctrl_tlv.us_event.chid",
+ FT_UINT8, BASE_DEC, NULL, 0x0,
+ "Upstream Channel ID", HFILL}
+ },
+ {&hf_us_event_mask,
+ {".2 Upstream Status Event Enable Bitmask", "cmctrl_tlv.us_event.mask",
+ FT_BYTES, BASE_NONE, NULL, 0x0,
+ "Upstream Status Event Enable Bitmask", HFILL}
+ },
+ {&hf_cmctrl_tlv_event,
+ {"7 Override Non-Channel-Specific Events", "cmctrl_tlv.event",
+ FT_BYTES, BASE_NONE, NULL, 0x0,
+ "Override Non-Channel-Specific Events", HFILL}
+ },
+ };
+
+/* Setup protocol subtree array */
+ static gint *ett[] = {
+ &ett_cmctrl_tlv,
+ &ett_cmctrl_tlv_ds_event,
+ &ett_cmctrl_tlv_us_event,
+ };
+
+/* Register the protocol name and description */
+ proto_cmctrl_tlv = proto_register_protocol ("DOCSIS CM-CTRL TLV's",
+ "DOCSIS CM-CTRL TLVs", "cmctrl_tlv");
+
+/* Required function calls to register the header fields and subtrees used */
+ proto_register_field_array (proto_cmctrl_tlv, hf, array_length (hf));
+ proto_register_subtree_array (ett, array_length (ett));
+
+ register_dissector ("cmctrl_tlv", dissect_cmctrl_tlv, proto_cmctrl_tlv);
+}
+
+/* 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_cmctrl_tlv (void)
+{
+ dissector_handle_t cmctrl_tlv_handle;
+
+ cmctrl_tlv_handle = find_dissector ("cmctrl_tlv");
+
+ dissector_add ("docsis", 0xFE, cmctrl_tlv_handle);
+}
+