aboutsummaryrefslogtreecommitdiffstats
path: root/epan/dissectors/packet-5co-legacy.c
diff options
context:
space:
mode:
Diffstat (limited to 'epan/dissectors/packet-5co-legacy.c')
-rw-r--r--epan/dissectors/packet-5co-legacy.c923
1 files changed, 923 insertions, 0 deletions
diff --git a/epan/dissectors/packet-5co-legacy.c b/epan/dissectors/packet-5co-legacy.c
new file mode 100644
index 0000000000..161828355e
--- /dev/null
+++ b/epan/dissectors/packet-5co-legacy.c
@@ -0,0 +1,923 @@
+/* packet-5co-legacy.c
+ * Routines for FiveCo's Legacy Register Access Protocol dissector
+ * Copyright 2021, Antoine Gardiol <antoine.gardiol@fiveco.ch>
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+/*
+ * This protocol allows access to FiveCo's Ethernet products registers with old legacy
+ * protocol. Product list can be found under https://www.fiveco.ch/bus-converter-products.html.
+ * Protocol description can be found (by example) in FMod-TCP xx manual that can be dowloaded from
+ * https://www.fiveco.ch/product-fmod-tcp-db.html.
+ * Note that this protocol is a question-answer protocol. It's header is composed of:
+ * - 16 bits type
+ * - 16 bits frame id
+ * - 16 bits length of parameters (n)
+ * - n bytes of parameters (depends upon packet type)
+ * - 16 bits IP like checksum
+ *
+ * This build-in dissector is replacing a plugin dissector available from Wireshark 1.8.
+ */
+
+#include <config.h>
+#include <epan/packet.h>
+#include <epan/proto_data.h>
+#include <string.h>
+#include "packet-tcp.h"
+
+/* Prototypes */
+void proto_reg_handoff_FiveCoLegacy(void);
+void proto_register_FiveCoLegacy(void);
+
+static dissector_handle_t FiveCoLegacy_handle;
+
+/****************************************************************************/
+/* Definition declaration */
+/****************************************************************************/
+
+// Protocol header length and frame minimum length
+#define FIVECO_LEGACY_HEADER_LENGTH 6
+#define FIVECO_LEGACY_MIN_LENGTH FIVECO_LEGACY_HEADER_LENGTH + 2 // Checksum is 16 bits
+
+#define PSNAME "5co-legacy"
+
+/* Global sample ports preferences */
+#define FIVECO_PORT1 8010 /* TCP port of the FiveCo protocol */
+#define FIVECO_PORT2 8004 /* TCP port of the FiveCo protocol for web page upload */
+#define FIVECO_UDP_PORT1 7010 /* UDP port of the FiveCo protocol */
+
+/* 16 bits type known available functions */
+enum fiveco_functions
+{
+ I2C_READ = 0x0001,
+ I2C_WRITE,
+ I2C_READ_ANSWER,
+ I2C_WRITE_ANSWER,
+ I2C_SCAN,
+ I2C_SCAN_ANSWER,
+ I2C_READ_WRITE_ACK,
+ I2C_READ_WRITE_ACK_ANSWER,
+ I2C_READ_WRITE_ACK_ERROR,
+ READ_REGISTER = 0x0021,
+ WRITE_REGISTER,
+ READ_REGISTER_ANSWER,
+ WRITE_REGISTER_ANSWER,
+ WRITE_REGISTER_QUIET,
+ EASY_IP_ADDRESS_CONFIG = 0x002A,
+ EASY_IP_ADDRESS_CONFIG_ANSWER,
+ FLASH_AREA_ERASE = 0x0031,
+ FLASH_AREA_LOAD,
+ FLASH_AREA_ANSWER
+};
+
+/* Forward references to functions */
+static guint16
+checksum_fiveco(tvbuff_t * byte_tab, guint16 start_offset, guint16 size);
+static gint fiveco_hash_equal(gconstpointer v, gconstpointer w);
+
+/* Register decoding functions prototypes */
+static void dispType( gchar *result, guint32 type);
+static void dispVersion( gchar *result, guint32 type);
+static void dispMAC( gchar *result, guint64 type);
+static void dispIP( gchar *result, guint32 type);
+static void dispMask( gchar *result, guint32 type);
+static void dispTimeout( gchar *result, guint32 type);
+
+/* Initialize the protocol and registered fields */
+static int proto_FiveCoLegacy; /* Wireshark ID of the FiveCo protocol */
+
+/* static dissector_handle_t data_handle = NULL; */
+static gint hf_fiveco_header; /* The following hf_* variables are used to hold the Wireshark IDs of */
+static gint hf_fiveco_fct; /* our header fields; they are filled out when we call */
+static gint hf_fiveco_id; /* proto_register_field_array() in proto_register_fiveco() */
+static gint hf_fiveco_length;
+static gint hf_fiveco_data;
+static gint hf_fiveco_cks;
+static gint hf_fiveco_i2cadd;
+static gint hf_fiveco_i2c2write;
+static gint hf_fiveco_i2cwrite;
+static gint hf_fiveco_i2c2read;
+static gint hf_fiveco_i2c2scan;
+static gint hf_fiveco_i2canswer;
+static gint hf_fiveco_i2cwriteanswer;
+static gint hf_fiveco_i2cscaned;
+static gint hf_fiveco_i2cerror;
+static gint hf_fiveco_i2cack;
+static gint hf_fiveco_regread;
+static gint hf_fiveco_regreadunknown;
+static gint hf_fiveco_regreaduk;
+static gint hf_fiveco_EasyIPMAC;
+static gint hf_fiveco_EasyIPIP;
+static gint hf_fiveco_EasyIPSM;
+
+static gint ett_fiveco_header; /* These are the ids of the subtrees that we may be creating */
+static gint ett_fiveco_data; /* for the header fields. */
+static gint ett_fiveco;
+static gint ett_fiveco_checksum;
+
+/* Constants declaration */
+static const value_string packettypenames[] = {
+ {I2C_READ, "I2C Read (deprecated)"},
+ {I2C_READ_ANSWER, "I2C Read Answer (deprecated)"},
+ {I2C_WRITE, "I2C Write (deprecated)"},
+ {I2C_WRITE_ANSWER, "I2C Write Answer (deprecated)"},
+ {I2C_SCAN, "I2C Scan"},
+ {I2C_SCAN_ANSWER, "I2C Scan Answer"},
+ {I2C_READ_WRITE_ACK, "I2C Read and write with ack"},
+ {I2C_READ_WRITE_ACK_ANSWER, "I2C Read and write with ack Answer"},
+ {I2C_READ_WRITE_ACK_ERROR, "I2C Read and write error"},
+ {READ_REGISTER, "Read register"},
+ {READ_REGISTER_ANSWER, "Read register Answer"},
+ {WRITE_REGISTER, "Write register"},
+ {WRITE_REGISTER_ANSWER, "Write register Answer"},
+ {WRITE_REGISTER_QUIET, "Write register (no answer wanted)"},
+ {EASY_IP_ADDRESS_CONFIG, "Easy IP address config"},
+ {EASY_IP_ADDRESS_CONFIG_ANSWER, "Easy IP address config Acknowledge"},
+ {FLASH_AREA_ERASE, "Flash area Erase"},
+ {FLASH_AREA_LOAD, "Flash area Upload"},
+ {FLASH_AREA_ANSWER, "Flash area Answer"},
+ {0, NULL}};
+
+/* Conversation request key structure */
+typedef struct
+{
+ guint32 conversation;
+ guint64 unInternalID;
+ guint16 usExpCmd;
+} FCOSConvRequestKey;
+
+/* Conversation request value structure */
+typedef struct
+{
+ guint16 usParaLen;
+ guint16 isReplied;
+ guint8 *pDataBuffer;
+} FCOSConvRequestVal;
+
+/* Conversation hash tables */
+static wmem_map_t *FiveCo_requests_hash = NULL;
+
+/* Internal unique ID (used to match answer with question
+ since some software set always 0 as packet ID in protocol header)
+*/
+static guint64 g_unInternalID = 0;
+
+/* Register definition structure (used to detect known registers when it is possible) */
+typedef struct
+{
+ guint32 unValue; // Register address
+ guint32 unSize; // Register size (in bytes)
+ const char *name; // Register name
+ const char *abbrev; // Abbreviation for header fill
+ const enum ftenum ft; // Field type
+ gint nsWsHeaderID; // Wireshark ID for header fill
+ const void *pFct; // Conversion function
+} FCOSRegisterDef;
+
+/* Known (common on every product) registers */
+static FCOSRegisterDef aRegisters[] = {
+ {0x00, 4, "Register Type/Model", "5co_legacy.RegTypeModel", FT_UINT32, -1, CF_FUNC(dispType)},
+ {0x01, 4, "Register Version", "5co_legacy.RegVersion", FT_UINT32, -1, CF_FUNC(dispVersion)},
+ {0x02, 0, "Function Reset device", "5co_legacy.RegReset", FT_NONE, -1, NULL},
+ {0x03, 0, "Function Save user parameters", "5co_legacy.RegSave", FT_NONE, -1, NULL},
+ {0x04, 0, "Function Restore user parameters", "5co_legacy.RegRestore", FT_NONE, -1, NULL},
+ {0x05, 0, "Function Restore factory parameters", "5co_legacy.RegRestoreFact", FT_NONE, -1, NULL},
+ {0x06, 0, "Function Save factory parameters", "5co_legacy.SaveFact", FT_NONE, -1, NULL},
+ {0x07, 0, "Register unknown", "5co_legacy.RegUnknown07", FT_NONE, -1, NULL},
+ {0x08, 0, "Register unknown", "5co_legacy.RegUnknown08", FT_NONE, -1, NULL},
+ {0x09, 0, "Register unknown", "5co_legacy.RegUnknown09", FT_NONE, -1, NULL},
+ {0x0A, 0, "Register unknown", "5co_legacy.RegUnknown0A", FT_NONE, -1, NULL},
+ {0x0B, 0, "Register unknown", "5co_legacy.RegUnknown0B", FT_NONE, -1, NULL},
+ {0x0C, 0, "Register unknown", "5co_legacy.RegUnknown0C", FT_NONE, -1, NULL},
+ {0x0D, 0, "Register unknown", "5co_legacy.RegUnknown0D", FT_NONE, -1, NULL},
+ {0x0E, 0, "Register unknown", "5co_legacy.RegUnknown0E", FT_NONE, -1, NULL},
+ {0x0F, 0, "Register unknown", "5co_legacy.RegUnknown0F", FT_NONE, -1, NULL},
+ {0x10, 4, "Register Communication options", "5co_legacy.RegComOption", FT_UINT32, -1, NULL},
+ {0x11, 6, "Register Ethernet MAC Address", "5co_legacy.RegMAC", FT_UINT48, -1, CF_FUNC(dispMAC)},
+ {0x12, 4, "Register IP Address", "5co_legacy.RegIPAdd", FT_UINT32, -1, CF_FUNC(dispIP)},
+ {0x13, 4, "Register IP Mask", "5co_legacy.RegIPMask", FT_UINT32, -1, CF_FUNC(dispMask)},
+ {0x14, 1, "Register TCP Timeout", "5co_legacy.RegTCPTimeout", FT_UINT8, -1, CF_FUNC(dispTimeout)},
+ {0x15, 16, "Register Module name", "5co_legacy.RegName", FT_STRING, -1, NULL}};
+
+ /* List of static header fields */
+static hf_register_info hf_base[] = {
+ {&hf_fiveco_header, {"Header", "5co_legacy.header", FT_NONE, BASE_NONE, NULL, 0x0, "Header of the packet", HFILL}},
+ {&hf_fiveco_fct, {"Function", "5co_legacy.fct", FT_UINT16, BASE_HEX, VALS(packettypenames), 0x0, "Function type", HFILL}},
+ {&hf_fiveco_id, {"Frame ID", "5co_legacy.id", FT_UINT16, BASE_DEC, NULL, 0x0, "Packet ID", HFILL}},
+ {&hf_fiveco_length, {"Data length", "5co_legacy.length", FT_UINT16, BASE_DEC, NULL, 0x0, "Parameters length of the packet", HFILL}},
+ {&hf_fiveco_data, {"Data", "5co_legacy.data", FT_NONE, BASE_NONE, NULL, 0x0, "Data (parameters)", HFILL}},
+ {&hf_fiveco_cks, {"Checksum", "5co_legacy.checksum", FT_UINT16, BASE_HEX, NULL, 0x0, "Checksum of the packet", HFILL}},
+ {&hf_fiveco_i2cadd, {"I2C Address", "5co_legacy.i2cadd", FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL}},
+ {&hf_fiveco_i2c2write, {"I2C number of bytes to write", "5co_legacy.i2c2write", FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL}},
+ {&hf_fiveco_i2cwrite, {"I2C bytes to write", "5co_legacy.i2cwrite", FT_NONE, BASE_NONE, NULL, 0x0, NULL, HFILL}},
+ {&hf_fiveco_i2c2read, {"I2C number of bytes to read", "5co_legacy.i2c2read", FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL}},
+ {&hf_fiveco_i2canswer, {"I2C bytes read", "5co_legacy.i2cread", FT_NONE, BASE_NONE, NULL, 0x0, NULL, HFILL}},
+ {&hf_fiveco_i2cwriteanswer, {"I2C bytes write", "5co_legacy.i2writeanswer", FT_NONE, BASE_NONE, NULL, 0x0, NULL, HFILL}},
+ {&hf_fiveco_i2cack, {"I2C ack state", "5co_legacy.i2cack", FT_UINT8, BASE_HEX, NULL, 0x0, NULL, HFILL}},
+ {&hf_fiveco_i2c2scan, {"I2C addresses to scan", "5co_legacy.i2c2scan", FT_NONE, BASE_NONE, NULL, 0x0, NULL, HFILL}},
+ {&hf_fiveco_i2cscaned, {"I2C addresses present", "5co_legacy.i2cscaned", FT_NONE, BASE_NONE, NULL, 0x0, NULL, HFILL}},
+ {&hf_fiveco_i2cerror, {"I2C error", "5co_legacy.i2cerror", FT_NONE, BASE_NONE, NULL, 0x0, NULL, HFILL}},
+ {&hf_fiveco_regread, {"Read", "5co_legacy.regread", FT_NONE, BASE_NONE, NULL, 0x0, NULL, HFILL}},
+ {&hf_fiveco_regreadunknown, {"Read Register unknown", "5co_legacy.hf_fiveco_regreadunknown", FT_NONE, BASE_NONE, NULL, 0x0, NULL, HFILL}},
+ {&hf_fiveco_regreaduk, {"Data not decoded", "5co_legacy.regreaduk", FT_NONE, BASE_NONE, NULL, 0x0, "Data not decoded because there are unable to map to a known register", HFILL}},
+ {&hf_fiveco_EasyIPMAC, {"MAC address", "5co_legacy.EasyIPMAC", FT_ETHER, BASE_NONE, NULL, 0x0, NULL, HFILL}},
+ {&hf_fiveco_EasyIPIP, {"New IP address", "5co_legacy.EasyIPIP", FT_IPv4, BASE_NONE, NULL, 0x0, NULL, HFILL}},
+ {&hf_fiveco_EasyIPSM, {"New subnet mask", "5co_legacy.EasyIPSM", FT_IPv4, BASE_NONE, NULL, 0x0, NULL, HFILL}}
+ };
+
+/*****************************************************************************/
+/* Code to actually dissect the packets */
+/* Callback function for reassembled packet */
+/*****************************************************************************/
+static int
+dissect_FiveCoLegacy(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void *data _U_)
+{
+ guint16 checksum_cal, checksum_rx;
+ guint16 i, j, y;
+ guint16 tcp_data_offset = 0;
+ guint32 tcp_data_length = 0;
+ guint16 header_type = 0;
+ guint16 header_id = 0;
+ guint16 header_data_length = 0;
+ guint8 data_i2c_length = 0;
+ proto_item *fiveco_item = NULL;
+ proto_item *fiveco_header_item = NULL;
+ proto_item *fiveco_data_item = NULL;
+ proto_tree *fiveco_tree = NULL;
+ proto_tree *fiveco_header_tree = NULL;
+ proto_tree *fiveco_data_tree = NULL;
+ conversation_t *conversation;
+ gboolean isRequest = FALSE;
+ guint64 *pulInternalID = NULL;
+ FCOSConvRequestKey requestKey, *pNewRequestKey;
+ FCOSConvRequestVal *pRequestVal = NULL;
+ tvbuff_t *pRequestTvb = NULL;
+ guint8 ucAdd, ucBytesToWrite, ucBytesToRead;
+ guint8 ucRegAdd, ucRegSize;
+ guint32 unOffset;
+ guint32 unSize;
+
+ /* Load protocol payload length (including checksum) */
+ tcp_data_length = tvb_captured_length(tvb);
+ if (tcp_data_length < FIVECO_LEGACY_MIN_LENGTH) // Check checksum presence
+ return 0;
+
+ /* Display fiveco in protocol column */
+ col_set_str(pinfo->cinfo, COL_PROTOCOL, PSNAME);
+ /* Clear out stuff in the info column */
+ col_clear(pinfo->cinfo, COL_INFO);
+
+ /* Look for all future TCP conversations between the
+ * requestiong server and the FiveCo device using the
+ * same src & dest addr and ports.
+ */
+ conversation = find_or_create_conversation(pinfo);
+ requestKey.conversation = conversation->conv_index;
+
+ /* Loop because several fiveco packets can be present in one TCP packet */
+ while (tcp_data_offset < tcp_data_length) {
+
+ /* Check that header type is correct */
+ header_type = tvb_get_ntohs(tvb, tcp_data_offset + 0);
+ if (try_val_to_str(header_type, packettypenames) == NULL)
+ return 0;
+
+ /* Read packet ID */
+ header_id = tvb_get_ntohs(tvb, tcp_data_offset + 2);
+
+ /* Check that there's enough data versus prot data header_data_length */
+ header_data_length = tvb_get_ntohs(tvb, tcp_data_offset + 4);
+ if (header_data_length > tcp_data_length - tcp_data_offset - 8) {
+ return 0;
+ }
+
+ /* Get/Set internal ID for this packet number */
+ pulInternalID = (guint64 *)p_get_proto_data(wmem_file_scope(), pinfo, proto_FiveCoLegacy, pinfo->num);
+ /* If internal ID is not set (null), create it */
+ if (!pulInternalID)
+ {
+ /* If it is a new request, increment internal ID */
+ if ((header_type == I2C_READ) || (header_type == I2C_WRITE) || (header_type == I2C_SCAN) ||
+ (header_type == I2C_READ_WRITE_ACK) || (header_type == READ_REGISTER) || (header_type == WRITE_REGISTER))
+ {
+ isRequest = TRUE;
+ g_unInternalID++; // Increment unique request ID and record it in the new request
+ /* Note: Since some software do not increment packet id located in frame header
+ we use an internal ID to match answers to request. */
+ }
+ pulInternalID = wmem_new(wmem_file_scope(), guint64);
+ *pulInternalID = g_unInternalID;
+ p_add_proto_data(wmem_file_scope(), pinfo, proto_FiveCoLegacy, pinfo->num, pulInternalID);
+ }
+
+ /* Get info about the request */
+ requestKey.usExpCmd = header_type;
+ requestKey.unInternalID = *pulInternalID;
+ pRequestVal = (FCOSConvRequestVal *)wmem_map_lookup(FiveCo_requests_hash, &requestKey);
+ if ((!pinfo->fd->visited) && (!pRequestVal) && (isRequest))
+ {
+ /* If unknown and if it is a request, allocate new hash element that we want to handle later in answer */
+ pNewRequestKey = wmem_new(wmem_file_scope(), FCOSConvRequestKey);
+ *pNewRequestKey = requestKey;
+ pNewRequestKey->unInternalID = g_unInternalID;
+ switch (header_type)
+ {
+ case I2C_READ:
+ pNewRequestKey->usExpCmd = I2C_READ_ANSWER;
+ break;
+ case I2C_WRITE:
+ pNewRequestKey->usExpCmd = I2C_WRITE_ANSWER;
+ break;
+ case I2C_SCAN:
+ pNewRequestKey->usExpCmd = I2C_SCAN_ANSWER;
+ break;
+ case I2C_READ_WRITE_ACK:
+ pNewRequestKey->usExpCmd = I2C_READ_WRITE_ACK_ANSWER;
+ break;
+ case READ_REGISTER:
+ pNewRequestKey->usExpCmd = READ_REGISTER_ANSWER;
+ break;
+ }
+
+ pRequestVal = wmem_new(wmem_file_scope(), FCOSConvRequestVal);
+ pRequestVal->usParaLen = header_data_length;
+ pRequestVal->isReplied = FALSE;
+ pRequestVal->pDataBuffer = (guint8 *)wmem_alloc(wmem_file_scope(), header_data_length);
+ tvb_memcpy(tvb, pRequestVal->pDataBuffer, tcp_data_offset + 6, header_data_length);
+
+ wmem_map_insert(FiveCo_requests_hash, pNewRequestKey, pRequestVal);
+ }
+
+ if (pRequestVal) {
+ pRequestTvb = tvb_new_child_real_data(tvb, pRequestVal->pDataBuffer, pRequestVal->usParaLen, pRequestVal->usParaLen);
+ }
+
+ /* Compute checksum of the packet and read one received */
+ checksum_cal = checksum_fiveco(tvb, tcp_data_offset, header_data_length + 6);
+ checksum_rx = tvb_get_ntohs(tvb, tcp_data_offset + header_data_length + 6);
+
+ /* Add text to info column */
+ /* If the offset != 0 (not first fiveco frame in tcp packet) add a comma in info column */
+ if (tcp_data_offset != 0)
+ {
+ col_append_fstr(pinfo->cinfo, COL_INFO, ", %s ID=%d Len=%d",
+ val_to_str(header_type, packettypenames, "Unknown Type:0x%02x"), header_id, header_data_length);
+ }
+ else
+ {
+ col_append_fstr(pinfo->cinfo, COL_INFO, "%s ID=%d Len=%d",
+ val_to_str(header_type, packettypenames, "Unknown Type:0x%02x"), header_id, header_data_length);
+ }
+
+ if (checksum_rx != checksum_cal)
+ {
+ col_append_str(pinfo->cinfo, COL_INFO, " [BAD CHECKSUM !!]");
+ }
+
+ /* Add FiveCo protocol in tree (after TCP or UDP entry) */
+ fiveco_item = proto_tree_add_item(tree, proto_FiveCoLegacy, tvb, tcp_data_offset + 0,
+ header_data_length + 8, ENC_NA); /* Add a new entry inside tree display */
+ proto_item_append_text(fiveco_item, " (%s)", val_to_str(header_type, packettypenames, "Unknown Type:0x%02x"));
+
+ /* Add fiveco Protocol tree and sub trees for Header, Data and Checksum */
+ fiveco_tree = proto_item_add_subtree(fiveco_item, ett_fiveco); // FiveCo prot tree
+ fiveco_header_item = proto_tree_add_item(fiveco_tree, hf_fiveco_header,
+ tvb, tcp_data_offset + 0, 6, ENC_NA); // Header tree
+ fiveco_header_tree = proto_item_add_subtree(fiveco_header_item, ett_fiveco_header);
+ proto_tree_add_item(fiveco_header_tree, hf_fiveco_fct,
+ tvb, tcp_data_offset + 0, 2, ENC_BIG_ENDIAN); // Packet type (function) in Header
+ proto_tree_add_item(fiveco_header_tree, hf_fiveco_id,
+ tvb, tcp_data_offset + 2, 2, ENC_BIG_ENDIAN); // Packet ID in Header
+ proto_tree_add_item(fiveco_header_tree, hf_fiveco_length,
+ tvb, tcp_data_offset + 4, 2, ENC_BIG_ENDIAN); // Length of para in Header
+
+ tcp_data_offset += 6; // put offset on start of data (parameters)
+
+ // If there are parameters (data) in packet, display them in data sub tree
+ if (header_data_length > 0)
+ {
+ fiveco_data_item = proto_tree_add_item(fiveco_tree, hf_fiveco_data, tvb, tcp_data_offset,
+ header_data_length, ENC_NA); // Data tree
+ fiveco_data_tree = proto_item_add_subtree(fiveco_data_item, ett_fiveco_data);
+ switch (header_type)
+ {
+ case I2C_READ:
+ case I2C_READ_WRITE_ACK:
+ i = 0;
+ while (i < header_data_length)
+ {
+ proto_tree_add_item(fiveco_data_tree, hf_fiveco_i2cadd, tvb, tcp_data_offset + i, 1, ENC_BIG_ENDIAN);
+ i += 1;
+ data_i2c_length = tvb_get_guint8(tvb, tcp_data_offset + i);
+ proto_tree_add_item(fiveco_data_tree, hf_fiveco_i2c2write, tvb, tcp_data_offset + i, 1, ENC_BIG_ENDIAN);
+ i += 1;
+ fiveco_data_item = proto_tree_add_item(fiveco_data_tree, hf_fiveco_i2cwrite,
+ tvb, tcp_data_offset + i, data_i2c_length, ENC_NA);
+ proto_item_append_text(fiveco_data_item, ": ");
+ for (j = 0; j < data_i2c_length; j++)
+ {
+ proto_item_append_text(fiveco_data_item, "0x%.2X ",
+ tvb_get_guint8(tvb, tcp_data_offset + i));
+ i += 1;
+ }
+ proto_tree_add_item(fiveco_data_tree, hf_fiveco_i2c2read, tvb, tcp_data_offset + i, 1, ENC_BIG_ENDIAN);
+ i += 1;
+ }
+ break;
+ case I2C_WRITE:
+ i = 0;
+ while (i < header_data_length)
+ {
+ proto_tree_add_item(fiveco_data_tree, hf_fiveco_i2cadd, tvb, tcp_data_offset + i, 1, ENC_BIG_ENDIAN);
+ i += 1;
+ data_i2c_length = tvb_get_guint8(tvb, tcp_data_offset + i);
+ proto_tree_add_item(fiveco_data_tree, hf_fiveco_i2c2write, tvb, tcp_data_offset + i, 1, ENC_BIG_ENDIAN);
+ i += 1;
+ fiveco_data_item = proto_tree_add_item(fiveco_data_tree, hf_fiveco_i2cwrite,
+ tvb, tcp_data_offset + i, data_i2c_length, ENC_NA);
+ proto_item_append_text(fiveco_data_item, ": ");
+ for (j = 0; j < data_i2c_length; j++)
+ {
+ proto_item_append_text(fiveco_data_item, "0x%.2X ",
+ tvb_get_guint8(tvb, tcp_data_offset + i));
+ i += 1;
+ }
+ }
+ break;
+ case I2C_SCAN:
+ fiveco_data_item = proto_tree_add_item(fiveco_data_tree, hf_fiveco_i2c2scan,
+ tvb, tcp_data_offset + 0, header_data_length, ENC_NA);
+ proto_item_append_text(fiveco_data_item, ": ");
+ // If specific address exists in packet, display them
+ for (i = 0; i < header_data_length; i++)
+ {
+ proto_item_append_text(fiveco_data_item, "0x%.2X ",
+ tvb_get_guint8(tvb, tcp_data_offset + i));
+ }
+ break;
+ case I2C_SCAN_ANSWER:
+ fiveco_data_item = proto_tree_add_item(fiveco_data_tree, hf_fiveco_i2cscaned,
+ tvb, tcp_data_offset + 0, header_data_length, ENC_NA);
+ proto_item_append_text(fiveco_data_item, ": ");
+ // Display slave address presents in answer
+ for (i = 0; i < header_data_length; i++)
+ {
+ proto_item_append_text(fiveco_data_item, "0x%.2X ",
+ tvb_get_guint8(tvb, tcp_data_offset + i));
+ }
+ break;
+ case I2C_READ_WRITE_ACK_ERROR:
+ fiveco_data_item = proto_tree_add_item(fiveco_data_tree, hf_fiveco_i2cerror,
+ tvb, tcp_data_offset + 0, header_data_length, ENC_NA);
+ proto_item_append_text(fiveco_data_item, ": ");
+ proto_item_append_text(fiveco_data_item, "0x%.2X ",
+ tvb_get_guint8(tvb, tcp_data_offset));
+ break;
+ case READ_REGISTER:
+ // List registers asked for read
+ for (i = 0; i < header_data_length; i++)
+ {
+ ucRegAdd = tvb_get_guint8(tvb, tcp_data_offset + i);
+ if ((ucRegAdd < array_length(aRegisters)) &&
+ (aRegisters[ucRegAdd].unValue == ucRegAdd))
+ {
+ fiveco_data_item = proto_tree_add_item(fiveco_data_tree, hf_fiveco_regread,
+ tvb, tcp_data_offset + i, 0, ENC_NA);
+ proto_item_append_text(fiveco_data_item, " %s", aRegisters[ucRegAdd].name);
+ }
+ else
+ {
+ fiveco_data_item = proto_tree_add_item(fiveco_data_tree, hf_fiveco_regreadunknown,
+ tvb, tcp_data_offset + i, 0, ENC_NA);
+ }
+ proto_item_append_text(fiveco_data_item, " (0x%.2X)", ucRegAdd);
+ }
+ break;
+ case WRITE_REGISTER:
+ case WRITE_REGISTER_QUIET:
+ // List register asked to write with data to fill in until an unknown one is found
+ for (i = tcp_data_offset; i < tcp_data_offset + header_data_length;)
+ {
+ ucRegAdd = tvb_get_guint8(tvb, i++);
+ // If register address is known & found
+ if ((ucRegAdd < array_length(aRegisters)) &&
+ (aRegisters[ucRegAdd].unValue == ucRegAdd))
+ {
+ ucRegSize = aRegisters[ucRegAdd].unSize;
+ // If a display function is defined, call it
+ if (aRegisters[ucRegAdd].pFct != NULL)
+ {
+ proto_tree_add_item(fiveco_data_tree, aRegisters[ucRegAdd].nsWsHeaderID,
+ tvb, i, ucRegSize, ENC_NA);
+ i += ucRegSize;
+ }
+ // else if register type is string, display it as string
+ else if (aRegisters[ucRegAdd].ft == FT_STRING)
+ {
+ fiveco_data_item = proto_tree_add_item(fiveco_data_tree,
+ aRegisters[ucRegAdd].nsWsHeaderID,
+ tvb, i, ucRegSize,
+ ENC_NA);
+ proto_item_append_text(fiveco_data_item, ": %s", tvb_format_text(pinfo->pool, tvb, i, ucRegSize));
+ i += ucRegSize;
+ }
+ // else display raw data in hex
+ else
+ {
+ fiveco_data_item = proto_tree_add_item(fiveco_data_tree, hf_fiveco_regread,
+ tvb, i, ucRegSize, ENC_NA);
+ proto_item_append_text(fiveco_data_item, " %s (Add: 0x%.2X, Size: %d bytes): ",
+ aRegisters[ucRegAdd].name, ucRegAdd, ucRegSize);
+ for (j = 0; j < ucRegSize; j++)
+ {
+ proto_item_append_text(fiveco_data_item, "0x%.2X ", tvb_get_guint8(tvb, i++));
+ }
+ }
+ }
+ // Else tell user that data cannot be interpreted
+ else
+ {
+ fiveco_data_item = proto_tree_add_item(fiveco_data_tree, hf_fiveco_regreaduk,
+ tvb, i, tcp_data_offset + header_data_length - i, ENC_NA);
+ proto_item_append_text(fiveco_data_item, " (Interpretation depends on product type)");
+ break;
+ }
+ }
+ break;
+ case EASY_IP_ADDRESS_CONFIG:
+ proto_tree_add_item(fiveco_data_tree, hf_fiveco_EasyIPMAC, tvb, tcp_data_offset + 0, 6, ENC_NA);
+ proto_tree_add_item(fiveco_data_tree, hf_fiveco_EasyIPIP, tvb, tcp_data_offset + 6, 4, ENC_BIG_ENDIAN);
+ proto_tree_add_item(fiveco_data_tree, hf_fiveco_EasyIPSM, tvb, tcp_data_offset + 10, 4, ENC_BIG_ENDIAN);
+ break;
+ case I2C_READ_ANSWER:
+ case I2C_WRITE_ANSWER:
+ case I2C_READ_WRITE_ACK_ANSWER:
+ if (pRequestVal)
+ {
+ if (pRequestVal->isReplied != 0)
+ {
+ proto_item_append_text(fiveco_data_item,
+ " WARNING : Answer already found ! Maybe packets ID not incremented.");
+ }
+ else
+ {
+ i = tcp_data_offset; // Answer index
+ y = 0; // Request index
+ while ((y < pRequestVal->usParaLen) && (i < tcp_data_offset + header_data_length))
+ {
+ // I2C address in first byte of request
+ ucAdd = tvb_get_guint8(pRequestTvb, y++);
+ // Read number of bytes to write
+ ucBytesToWrite = tvb_get_guint8(pRequestTvb, y);
+ // Skip number of bytes to write and those bytes
+ y += 1 + ucBytesToWrite;
+ // Read number of bytes to read
+ ucBytesToRead = tvb_get_guint8(pRequestTvb, y++);
+ if (ucBytesToRead > 0)
+ {
+ fiveco_data_item = proto_tree_add_item(fiveco_data_tree, hf_fiveco_i2canswer,
+ tvb, i, ucBytesToRead, ENC_NA);
+ proto_item_append_text(fiveco_data_item,
+ " from address %d (%d bytes written) : ",
+ ucAdd, ucBytesToWrite);
+ for (j = 0; j < ucBytesToRead; j++)
+ {
+ proto_item_append_text(fiveco_data_item, "0x%.2X ",
+ tvb_get_guint8(tvb, i++));
+ }
+ if (header_type == 0x08)
+ proto_tree_add_item(fiveco_data_tree, hf_fiveco_i2cack, tvb, i++, 1, ENC_BIG_ENDIAN);
+ }
+ else if (header_type == I2C_READ_WRITE_ACK_ANSWER)
+ {
+ // if it's an answer to a write but with ack, display it
+ fiveco_data_item = proto_tree_add_item(fiveco_data_tree,
+ hf_fiveco_i2cwriteanswer, tvb, i,
+ ucBytesToRead, ENC_NA);
+ proto_item_append_text(fiveco_data_item, " to address %d (%d bytes written)",
+ ucAdd, ucBytesToWrite);
+ proto_tree_add_item(fiveco_data_tree, hf_fiveco_i2cack, tvb, i++, 1, ENC_BIG_ENDIAN);
+ }
+ }
+ }
+ break;
+ }
+ else {
+ proto_item_append_text(fiveco_data_item, " (Interpretation depends on product type)");
+ }
+ break;
+ case READ_REGISTER_ANSWER:
+ if (pRequestVal)
+ {
+ if (pRequestVal->isReplied != 0)
+ {
+ proto_item_append_text(fiveco_data_item,
+ " WARNING : Answer already found ! Maybe packets ID not incremented.");
+ }
+ else
+ {
+ i = tcp_data_offset; // Answer index
+ y = 0; // Request index
+ // For each request stored in the last read request of the conversation
+ while ((y < pRequestVal->usParaLen) && (i < tcp_data_offset + header_data_length))
+ {
+ // Register address in first byte of request
+ ucRegAdd = tvb_get_guint8(pRequestTvb, y++);
+ // If register address is known & found in answer
+ if ((ucRegAdd < array_length(aRegisters)) &&
+ (aRegisters[ucRegAdd].unValue == ucRegAdd) &&
+ (ucRegAdd == tvb_get_guint8(tvb, i++)))
+ {
+ // Retrieve register size and display it with address
+ ucRegSize = aRegisters[ucRegAdd].unSize;
+ // If a display function is defined, call it
+ if (aRegisters[ucRegAdd].pFct != NULL)
+ {
+ proto_tree_add_item(fiveco_data_tree, aRegisters[ucRegAdd].nsWsHeaderID,
+ tvb, i, ucRegSize, ENC_NA);
+ i += ucRegSize;
+ }
+ // else if register type is string, display it as string
+ else if (aRegisters[ucRegAdd].ft == FT_STRING)
+ {
+ fiveco_data_item = proto_tree_add_item(fiveco_data_tree,
+ aRegisters[ucRegAdd].nsWsHeaderID,
+ tvb, i, ucRegSize,
+ ENC_NA);
+ proto_item_append_text(fiveco_data_item, ": %s", tvb_format_text(pinfo->pool, tvb, i, ucRegSize));
+ i += ucRegSize;
+ }
+ // else display raw data in hex
+ else
+ {
+ fiveco_data_item = proto_tree_add_item(fiveco_data_tree,
+ hf_fiveco_regread, tvb, i, ucRegSize, ENC_NA);
+ proto_item_append_text(fiveco_data_item,
+ " %s (Add: 0x%.2X, Size: %d bytes): ",
+ aRegisters[ucRegAdd].name, ucRegAdd, ucRegSize);
+ for (j = 0; j < ucRegSize; j++)
+ {
+ proto_item_append_text(fiveco_data_item,
+ "0x%.2X ", tvb_get_guint8(tvb, i++));
+ }
+ }
+ }
+ // Else tell user that data cannot be interpreted
+ else
+ {
+ fiveco_data_item = proto_tree_add_item(fiveco_data_tree,
+ hf_fiveco_regreaduk, tvb, i,
+ tcp_data_offset + header_data_length - i,
+ ENC_NA);
+ proto_item_append_text(fiveco_data_item,
+ " (Interpretation depends on product type)");
+ break;
+ }
+ }
+ }
+ }
+ break;
+ case FLASH_AREA_LOAD:
+ unOffset = tvb_get_guint24(tvb, tcp_data_offset, ENC_BIG_ENDIAN);
+ unSize = tvb_get_guint24(tvb, tcp_data_offset + 3, ENC_BIG_ENDIAN);
+ proto_item_append_text(fiveco_data_item,
+ " (%d bytes to load into flash at offset %d)", unSize, unOffset);
+ break;
+ case FLASH_AREA_ANSWER:
+ if ( header_data_length > 1 ) {
+ proto_item_append_text(fiveco_data_item, " (%s)", tvb_format_text(pinfo->pool, tvb, tcp_data_offset, header_data_length - 1));
+ }
+ break;
+
+ case WRITE_REGISTER_ANSWER:
+ case FLASH_AREA_ERASE:
+ case EASY_IP_ADDRESS_CONFIG_ANSWER:
+ proto_item_append_text(fiveco_data_item, " (ERROR: No data should be present with that packet type !!)");
+ break;
+
+ default:
+ proto_item_append_text(fiveco_data_item, " (Interpretation depends on product type)");
+ break;
+ }
+ }
+
+ // Checksum validation and sub tree
+ proto_tree_add_checksum(fiveco_tree, tvb, tcp_data_offset + header_data_length, hf_fiveco_cks, -1, NULL, NULL,
+ checksum_cal, ENC_BIG_ENDIAN, PROTO_CHECKSUM_VERIFY);
+
+ tcp_data_offset += header_data_length + 2 ; /* jump to next packet if exists */
+ } /*while (tcp_data_offset < tcp_data_length) */
+
+ return tvb_captured_length(tvb);
+}
+
+/*****************************************************************************/
+/* This function returns the calculated checksum (IP based) */
+/*****************************************************************************/
+static guint16 checksum_fiveco(tvbuff_t *byte_tab, guint16 start_offset, guint16 size)
+{
+ guint32 Sum = 0;
+ guint8 AddHighByte = 1;
+ guint32 ChecksumCalculated;
+ guint16 i;
+ guint8 temp;
+
+ for (i = 0; i < size; i++)
+ {
+ tvb_memcpy(byte_tab, (guint8 *)&temp, start_offset + i, 1);
+ if (AddHighByte)
+ {
+ Sum += (temp << 8) ^ 0xFF00;
+ AddHighByte = 0;
+ }
+ else
+ {
+ Sum += (temp) ^ 0x00FF;
+ AddHighByte = 1;
+ }
+ }
+
+ if (AddHighByte == 0)
+ Sum += 0xFF;
+
+ ChecksumCalculated = ((Sum >> 16) & 0xFFFF) + (Sum & 0xFFFF);
+ ChecksumCalculated = ((ChecksumCalculated >> 16) & 0xFFFF) + (ChecksumCalculated & 0xFFFF);
+ return (guint16)ChecksumCalculated;
+}
+
+/*****************************************************************************/
+/* Compute an unique hash value */
+/*****************************************************************************/
+static guint fiveco_hash(gconstpointer v)
+{
+ const FCOSConvRequestKey *key = (const FCOSConvRequestKey *)v;
+ guint val;
+
+ val = key->conversation + (((key->usExpCmd) & 0xFFFF) << 16) +
+ (key->unInternalID & 0xFFFFFFFF) + ((key->unInternalID >>32) & 0xFFFFFFFF);
+
+ return val;
+}
+
+/*****************************************************************************/
+/* Check hash equal */
+/*****************************************************************************/
+static gint fiveco_hash_equal(gconstpointer v, gconstpointer w)
+{
+ const FCOSConvRequestKey *v1 = (const FCOSConvRequestKey *)v;
+ const FCOSConvRequestKey *v2 = (const FCOSConvRequestKey *)w;
+
+ if (v1->conversation == v2->conversation &&
+ v1->usExpCmd == v2->usExpCmd &&
+ v1->unInternalID == v2->unInternalID)
+ {
+ return 1;
+ }
+ return 0;
+}
+
+/*****************************************************************************/
+/* Register the protocol with Wireshark.
+ *
+ * This format is required because a script is used to build the C function that
+ * calls all the protocol registration.
+ */
+/*****************************************************************************/
+void proto_register_FiveCoLegacy(void)
+{
+ /* Setup list of header fields (based on static table and specific table) */
+ static hf_register_info hf[array_length(hf_base) + array_length(aRegisters)];
+ for (guint32 i = 0; i < array_length(hf_base); i++) {
+ hf[i] = hf_base[i];
+ }
+ for (guint32 i = 0; i < array_length(aRegisters); i++) {
+ if (aRegisters[i].pFct != NULL){
+ hf_register_info hfx = { &(aRegisters[i].nsWsHeaderID),{aRegisters[i].name, aRegisters[i].abbrev, aRegisters[i].ft, BASE_CUSTOM, aRegisters[i].pFct, 0x0, NULL, HFILL}};
+ hf[array_length(hf_base) + i] = hfx;
+ } else {
+ hf_register_info hfx = { &(aRegisters[i].nsWsHeaderID),{aRegisters[i].name, aRegisters[i].abbrev, FT_NONE, BASE_NONE, NULL, 0x0, NULL, HFILL}};
+ hf[array_length(hf_base) + i] = hfx;
+ }
+ }
+
+ /* Setup protocol subtree array */
+ static gint *ett[] = {
+ &ett_fiveco_header,
+ &ett_fiveco_data,
+ &ett_fiveco,
+ &ett_fiveco_checksum};
+
+ /* Register the protocol name and description */
+ proto_FiveCoLegacy = proto_register_protocol("FiveCo's Legacy Register Access Protocol",
+ PSNAME, "5co_legacy");
+
+ /* Required function calls to register the header fields and subtrees */
+ proto_register_field_array(proto_FiveCoLegacy, hf, array_length(hf));
+ proto_register_subtree_array(ett, array_length(ett));
+
+ /* Register the dissector */
+ FiveCoLegacy_handle = register_dissector("5co_legacy", dissect_FiveCoLegacy,
+ proto_FiveCoLegacy);
+
+ FiveCo_requests_hash = wmem_map_new_autoreset(wmem_epan_scope(), wmem_file_scope(), fiveco_hash, fiveco_hash_equal);
+
+ /* Set preference callback to NULL since it is not used */
+ prefs_register_protocol(proto_FiveCoLegacy, NULL);
+}
+
+/* If this dissector uses sub-dissector registration add a registration routine.
+ * This exact format is required because a script is used to find these
+ * routines and create the code that calls these routines.
+ *
+ * Simpler form of proto_reg_handoff_FiveCoLegacy which can be used if there are
+ * no prefs-dependent registration function calls. */
+void proto_reg_handoff_FiveCoLegacy(void)
+{
+ static gboolean initialized = FALSE;
+
+ if (!initialized)
+ {
+ dissector_add_uint("tcp.port", FIVECO_PORT1, FiveCoLegacy_handle);
+ dissector_add_uint("tcp.port", FIVECO_PORT2, FiveCoLegacy_handle);
+ dissector_add_uint("udp.port", FIVECO_UDP_PORT1, FiveCoLegacy_handle);
+ initialized = TRUE;
+ }
+}
+
+/*****************************************************************************/
+/* Registers decoding functions */
+/*****************************************************************************/
+static void
+dispType( gchar *result, guint32 type)
+{
+ int nValueH = (type>>16) & 0xFFFF;
+ int nValueL = (type & 0xFFFF);
+ snprintf( result, 18, "%d.%d (%.4X.%.4X)", nValueH, nValueL, nValueH, nValueL);
+}
+
+static void
+dispVersion( gchar *result, guint32 version)
+{
+ if ((version & 0xFF000000) == 0)
+ {
+ int nValueH = (version>>16) & 0xFFFF;
+ int nValueL = (version & 0xFFFF);
+ snprintf( result, 11, "FW: %d.%d", nValueH, nValueL);
+ }
+ else
+ {
+ int nHWHigh = (version>>24) & 0xFF;
+ int nHWLow = (version>>16) & 0xFF;
+ int nFWHigh = (version>>8) & 0xFF;
+ int nFWLow = (version>>8) & 0xFF;
+ snprintf( result, 25, "HW: %d.%d / FW: %d.%d", nHWHigh, nHWLow, nFWHigh, nFWLow);
+ }
+}
+
+static void dispMAC( gchar *result, guint64 mac)
+{
+ guint8 *pData = (guint8*)(&mac);
+
+ snprintf( result, 18, "%.2X-%.2X-%.2X-%.2X-%.2X-%.2X", pData[5], pData[4], pData[3], pData[2],
+ pData[1], pData[0]);
+}
+
+static void dispIP( gchar *result, guint32 ip)
+{
+ guint8 *pData = (guint8*)(&ip);
+
+ snprintf( result, 15, "%d.%d.%d.%d", pData[3], pData[2], pData[1], pData[0]);
+}
+
+static void dispMask( gchar *result, guint32 mask)
+{
+ guint8 *pData = (guint8*)(&mask);
+
+ snprintf( result, 15, "%d.%d.%d.%d", pData[3], pData[2], pData[1], pData[0]);
+}
+
+static void dispTimeout( gchar *result, guint32 timeout)
+{
+ if (timeout != 0)
+ snprintf( result, 12, "%d secondes", timeout);
+ else
+ snprintf( result, 8, "Disabled");
+}
+
+/*
+ * 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:
+ */