summaryrefslogtreecommitdiffstats
path: root/epan/dissectors/packet-sinecap.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-sinecap.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-sinecap.c')
-rw-r--r--epan/dissectors/packet-sinecap.c208
1 files changed, 208 insertions, 0 deletions
diff --git a/epan/dissectors/packet-sinecap.c b/epan/dissectors/packet-sinecap.c
new file mode 100644
index 00000000..4bdc85f6
--- /dev/null
+++ b/epan/dissectors/packet-sinecap.c
@@ -0,0 +1,208 @@
+/* packet-sinecap.c
+ *
+ * Author: Nikolas Koesling, 2023 (nikolas@koesling.info)
+ * Description: Wireshark dissector for the SINEC AP protocol according to
+ * https://cache.industry.siemens.com/dl/files/274/22090274/att_83836/v1/447_840_840C_880_Computer_Link_General_Description.pdf
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#include "config.h"
+#include <epan/packet.h>
+
+#define PROTO_TAG_AP "SINEC-AP"
+
+/* Min. telegram length for heuristic check */
+#define TXP_MIN_TELEGRAM_LENGTH 22
+
+/* Wireshark ID of the AP1 protocol */
+static gint proto_ap = -1;
+
+static gint hf_ap_protoid = -1;
+static gint hf_ap_mpxadr = -1;
+static gint hf_ap_comcls = -1;
+static gint hf_ap_comcod = -1;
+static gint hf_ap_modfr1 = -1;
+static gint hf_ap_modfr2 = -1;
+static gint hf_ap_errcls = -1;
+static gint hf_ap_errcod = -1;
+static gint hf_ap_rosctr = -1;
+static gint hf_ap_sgsqnr = -1;
+static gint hf_ap_tactid = -1;
+static gint hf_ap_tasqnr = -1;
+static gint hf_ap_spare = -1;
+static gint hf_ap_pduref = -1;
+static gint hf_ap_pduid = -1;
+static gint hf_ap_pdulg = -1;
+static gint hf_ap_parlg = -1;
+static gint hf_ap_datlg = -1;
+
+static gint ett_ap = -1;
+
+static heur_dissector_list_t ap_heur_subdissector_list;
+
+static const value_string vs_comcls[] = {
+ {0x0, "ACK without data"},
+ {0x4, "Serial transfer"},
+ {0, NULL}
+};
+
+static const value_string vs_protid[] = {
+ {0x0, "SINEC AP 1.0"},
+ {0, NULL}
+};
+
+static gboolean
+dissect_ap(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree _U_, void *data _U_)
+{
+ /*----------------- Heuristic Checks - Begin */
+ /* 1) check for minimum length */
+ if (tvb_captured_length(tvb) < TXP_MIN_TELEGRAM_LENGTH)
+ return FALSE;
+
+ /* 2) protocol id == 0 */
+ if (tvb_get_guint8(tvb, 0) != 0)
+ return FALSE;
+ /*----------------- Heuristic Checks - End */
+
+ col_set_str(pinfo->cinfo, COL_PROTOCOL, PROTO_TAG_AP);
+ col_clear(pinfo->cinfo, COL_INFO);
+
+ guint8 comcls = tvb_get_guint8(tvb, 2);
+
+ gint offset = 16;
+ guint16 pdulg = tvb_get_guint16(tvb, offset, ENC_BIG_ENDIAN);
+ offset += 4;
+
+ guint16 datlg = tvb_get_guint16(tvb, offset, ENC_BIG_ENDIAN);
+ offset += 2;
+
+ ws_assert(offset == 22);
+
+ /* check pdu and data length */
+ if (pdulg != tvb_captured_length(tvb))
+ return FALSE;
+ if (datlg != tvb_captured_length(tvb) - 22)
+ return FALSE;
+
+ switch (comcls) {
+ case 0x0: {
+ // ack without data
+ col_append_fstr(pinfo->cinfo, COL_INFO, "%s", "ACK without data");
+ break;
+ }
+ case 0x4: {
+ // serial transfer
+ col_append_fstr(pinfo->cinfo, COL_INFO, "%s", "Serial transfer");
+ break;
+ }
+ default:
+ col_append_fstr(pinfo->cinfo, COL_INFO, "%s", "UNKNOWN command class");
+ }
+
+ proto_item *ap_item = proto_tree_add_item(tree, proto_ap, tvb, 0, -1, ENC_NA);
+ proto_tree *ap_tree = proto_item_add_subtree(ap_item, ett_ap);
+
+ offset = 0;
+ proto_tree_add_item(ap_tree, hf_ap_protoid, tvb, offset++, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(ap_tree, hf_ap_mpxadr, tvb, offset++, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(ap_tree, hf_ap_comcls, tvb, offset++, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(ap_tree, hf_ap_comcod, tvb, offset++, 1, ENC_BIG_ENDIAN);
+
+ switch (comcls) {
+ case 0x0: {
+ // ack without data
+ proto_tree_add_item(ap_tree, hf_ap_errcls, tvb, offset++, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(ap_tree, hf_ap_errcod, tvb, offset++, 1, ENC_BIG_ENDIAN);
+ break;
+ }
+ case 0x4: {
+ // serial transfer
+ proto_tree_add_item(ap_tree, hf_ap_modfr1, tvb, offset++, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(ap_tree, hf_ap_modfr2, tvb, offset++, 1, ENC_BIG_ENDIAN);
+ break;
+ }
+ default:
+ proto_tree_add_item(ap_tree, hf_ap_modfr1, tvb, offset++, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(ap_tree, hf_ap_modfr2, tvb, offset++, 1, ENC_BIG_ENDIAN);
+ }
+
+ proto_tree_add_item(ap_tree, hf_ap_rosctr, tvb, offset++, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(ap_tree, hf_ap_sgsqnr, tvb, offset++, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(ap_tree, hf_ap_tactid, tvb, offset++, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(ap_tree, hf_ap_tasqnr, tvb, offset++, 1, ENC_BIG_ENDIAN);
+ proto_tree_add_item(ap_tree, hf_ap_spare, tvb, offset, 2, ENC_BIG_ENDIAN);
+ offset += 2;
+ proto_tree_add_item(ap_tree, hf_ap_pduref, tvb, offset, 2, ENC_BIG_ENDIAN);
+ offset += 2;
+ proto_tree_add_item(ap_tree, hf_ap_pduid, tvb, offset, 2, ENC_BIG_ENDIAN);
+ offset += 2;
+ proto_tree_add_item(ap_tree, hf_ap_pdulg, tvb, offset, 2, ENC_BIG_ENDIAN);
+ offset += 2;
+ proto_tree_add_item(ap_tree, hf_ap_parlg, tvb, offset, 2, ENC_BIG_ENDIAN);
+ offset += 2;
+ proto_tree_add_item(ap_tree, hf_ap_datlg, tvb, offset, 2, ENC_BIG_ENDIAN);
+ offset += 2;
+
+ ws_assert(offset == 22);
+
+ if (tvb_reported_length_remaining(tvb, offset) > 0) {
+ struct tvbuff *next_tvb = tvb_new_subset_remaining(tvb, offset);
+ heur_dtbl_entry_t *hdtbl_entry;
+ if (!dissector_try_heuristic(ap_heur_subdissector_list, next_tvb, pinfo, tree, &hdtbl_entry, NULL)) {
+ call_data_dissector(next_tvb, pinfo, tree);
+ }
+ }
+
+ return TRUE;
+}
+
+void
+proto_register_ap(void)
+{
+ static hf_register_info hf[] = {
+ {&hf_ap_protoid, {"PROTID", "sinecap.protid", FT_UINT8, BASE_HEX, VALS(vs_protid), 0x0, "Protocol version", HFILL}},
+ {&hf_ap_mpxadr, {"MPXADR", "sinecap.mpxadr", FT_UINT8, BASE_HEX, NULL, 0x0, "Multiplex address", HFILL}},
+ {&hf_ap_comcls, {"COMCLS", "sinecap.comcls", FT_UINT8, BASE_HEX, VALS(vs_comcls), 0x0, "Command class", HFILL}},
+ {&hf_ap_comcod, {"COMCOD", "sinecap.comcod", FT_UINT8, BASE_HEX, NULL, 0x0, "Command code", HFILL}},
+ {&hf_ap_modfr1, {"MODFR1", "sinecap.modfr1", FT_UINT8, BASE_HEX, NULL, 0x0, "Modifier 1", HFILL}},
+ {&hf_ap_errcls, {"ERRCLS", "sinecap.errcls", FT_UINT8, BASE_HEX, NULL, 0x0, "Error class", HFILL}},
+ {&hf_ap_modfr2, {"MODFR2", "sinecap.modfr2", FT_UINT8, BASE_HEX, NULL, 0x0, "Modifier 2", HFILL}},
+ {&hf_ap_errcod, {"ERRCOD", "sinecap.errcod", FT_UINT8, BASE_HEX, NULL, 0x0, "Error code", HFILL}},
+ {&hf_ap_rosctr, {"ROSCTR", "sinecap.rosctr", FT_UINT8, BASE_HEX, NULL, 0x0, "Remote operating service", HFILL}},
+ {&hf_ap_sgsqnr, {"SGSQNR", "sinecap.sgsqnr", FT_UINT8, BASE_HEX_DEC, NULL, 0x0, "Segment sequence number", HFILL}},
+ {&hf_ap_tactid, {"TACTID", "sinecap.tactid", FT_UINT8, BASE_HEX, NULL, 0x0, "Transaction identifier", HFILL}},
+ {&hf_ap_tasqnr, {"TASQNR", "sinecap.tasqnr", FT_UINT8, BASE_HEX, NULL, 0x0, "Transaction sequence number", HFILL}},
+ {&hf_ap_spare, {"SPARE", "sinecap.spare", FT_UINT16, BASE_HEX, NULL, 0x0, "Free space", HFILL}},
+ {&hf_ap_pduref, {"PDUREF", "sinecap.pduref", FT_UINT16, BASE_HEX, NULL, 0x0, "Protocol Data Unit reference", HFILL}},
+ {&hf_ap_pduid, {"PDUID", "sinecap.pduid", FT_UINT16, BASE_HEX, NULL, 0x0, "Protocol Data Unit identifier", HFILL}},
+ {&hf_ap_pdulg, {"PDULG", "sinecap.pdulg", FT_UINT16, BASE_HEX_DEC, NULL, 0x0, "Protocol Data Unit length", HFILL}},
+ {&hf_ap_parlg, {"PARLG", "sinecap.parlg", FT_UINT16, BASE_HEX_DEC, NULL, 0x0, "Parameter length", HFILL}},
+ {&hf_ap_datlg, {"DATLG", "sinecap.datlg", FT_UINT16, BASE_HEX_DEC, NULL, 0x0, "Data length", HFILL}},
+ };
+
+ proto_ap = proto_register_protocol (
+ "SINEC AP Telegram", /* name */
+ "SINEC AP", /* short name */
+ "sinecap" /* filter_name */
+ );
+
+ static gint *ett[] = {
+ &ett_ap,
+ };
+
+ proto_register_field_array(proto_ap, hf, array_length(hf));
+ proto_register_subtree_array(ett, array_length (ett));
+ ap_heur_subdissector_list = register_heur_dissector_list("sinecap", proto_ap);
+}
+
+void
+proto_reg_handoff_ap(void)
+{
+ heur_dissector_add("cotp", dissect_ap, "SINEC AP Telegram over COTP", "sinecap", proto_ap, HEURISTIC_ENABLE);
+ heur_dissector_add("cotp_is", dissect_ap, "SINEC AP Telegram over COTP", "sinecap_is", proto_ap, HEURISTIC_ENABLE);
+}