summaryrefslogtreecommitdiffstats
path: root/epan/decode_as.c
diff options
context:
space:
mode:
Diffstat (limited to 'epan/decode_as.c')
-rw-r--r--epan/decode_as.c550
1 files changed, 550 insertions, 0 deletions
diff --git a/epan/decode_as.c b/epan/decode_as.c
new file mode 100644
index 00000000..72afd0c0
--- /dev/null
+++ b/epan/decode_as.c
@@ -0,0 +1,550 @@
+/* decode_as.c
+ * Routines for dissector Decode As handlers
+ *
+ * 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 <glib.h>
+
+#include "decode_as.h"
+#include "packet.h"
+#include "prefs.h"
+#include "prefs-int.h"
+#include "wsutil/file_util.h"
+#include "wsutil/filesystem.h"
+#include "epan/dissectors/packet-dcerpc.h"
+#include <stdio.h>
+#include <stdlib.h>
+#include <errno.h>
+#include <wsutil/ws_assert.h>
+
+GList *decode_as_list = NULL;
+
+void register_decode_as(decode_as_t* reg)
+{
+ dissector_table_t decode_table;
+
+ /* Ensure valid functions */
+ ws_assert(reg->populate_list);
+ ws_assert(reg->reset_value);
+ ws_assert(reg->change_value);
+
+ decode_table = find_dissector_table(reg->table_name);
+ if (decode_table != NULL)
+ {
+ dissector_table_allow_decode_as(decode_table);
+ }
+
+ decode_as_list = g_list_prepend(decode_as_list, reg);
+}
+
+static void next_proto_prompt(packet_info *pinfo _U_, gchar *result)
+{
+ snprintf(result, MAX_DECODE_AS_PROMPT_LEN, "Next level protocol as");
+}
+
+static gpointer next_proto_value(packet_info *pinfo _U_)
+{
+ return 0;
+}
+
+static build_valid_func next_proto_values[] = { next_proto_value };
+static decode_as_value_t next_proto_da_values =
+ { next_proto_prompt, 1, next_proto_values };
+
+dissector_table_t register_decode_as_next_proto(int proto, const gchar *table_name, const gchar *ui_name, build_label_func label_func)
+{
+ decode_as_t *da;
+
+ dissector_table_t dt = register_dissector_table(table_name, ui_name, proto, FT_NONE, BASE_NONE);
+
+ da = wmem_new0(wmem_epan_scope(), decode_as_t);
+ da->name = wmem_strdup(wmem_epan_scope(), proto_get_protocol_filter_name(proto));
+ da->table_name = wmem_strdup(wmem_epan_scope(), table_name);
+ da->num_items = 1;
+ if (label_func == NULL)
+ {
+ da->values = &next_proto_da_values;
+ }
+ else
+ {
+ da->values = wmem_new(wmem_epan_scope(), decode_as_value_t);
+ da->values->label_func = label_func;
+ da->values->num_values = 1;
+ da->values->build_values = next_proto_values;
+ }
+ da->populate_list = decode_as_default_populate_list;
+ da->reset_value = decode_as_default_reset;
+ da->change_value = decode_as_default_change;
+
+ register_decode_as(da);
+ return dt;
+}
+
+struct decode_as_default_populate
+{
+ decode_as_add_to_list_func add_to_list;
+ gpointer ui_element;
+};
+
+static void
+decode_proto_add_to_list (const gchar *table_name, gpointer value, gpointer user_data)
+{
+ struct decode_as_default_populate* populate = (struct decode_as_default_populate*)user_data;
+ const gchar *dissector_description;
+ gint i;
+ dissector_handle_t handle;
+
+
+ handle = (dissector_handle_t)value;
+ dissector_description = dissector_handle_get_description(handle);
+
+ i = dissector_handle_get_protocol_index(handle);
+ if (i >= 0 && !proto_is_protocol_enabled(find_protocol_by_id(i)))
+ return;
+
+ populate->add_to_list(table_name, dissector_description, value, populate->ui_element);
+}
+
+void decode_as_default_populate_list(const gchar *table_name, decode_as_add_to_list_func add_to_list, gpointer ui_element)
+{
+ struct decode_as_default_populate populate;
+
+ populate.add_to_list = add_to_list;
+ populate.ui_element = ui_element;
+
+ dissector_table_foreach_handle(table_name, decode_proto_add_to_list, &populate);
+}
+
+gboolean decode_as_default_reset(const gchar *name, gconstpointer pattern)
+{
+ switch (get_dissector_table_selector_type(name)) {
+ case FT_UINT8:
+ case FT_UINT16:
+ case FT_UINT24:
+ case FT_UINT32:
+ dissector_reset_uint(name, GPOINTER_TO_UINT(pattern));
+ return TRUE;
+ case FT_NONE:
+ dissector_reset_payload(name);
+ return TRUE;
+ case FT_STRING:
+ case FT_STRINGZ:
+ case FT_UINT_STRING:
+ case FT_STRINGZPAD:
+ case FT_STRINGZTRUNC:
+ dissector_reset_string(name, (!pattern)?"":(const gchar *) pattern);
+ return TRUE;
+ default:
+ return FALSE;
+ };
+
+ return TRUE;
+}
+
+gboolean decode_as_default_change(const gchar *name, gconstpointer pattern, gconstpointer handle, const gchar *list_name _U_)
+{
+ const dissector_handle_t dissector = (const dissector_handle_t)handle;
+ switch (get_dissector_table_selector_type(name)) {
+ case FT_UINT8:
+ case FT_UINT16:
+ case FT_UINT24:
+ case FT_UINT32:
+ dissector_change_uint(name, GPOINTER_TO_UINT(pattern), dissector);
+ return TRUE;
+ case FT_NONE:
+ dissector_change_payload(name, dissector);
+ return TRUE;
+ case FT_STRING:
+ case FT_STRINGZ:
+ case FT_UINT_STRING:
+ case FT_STRINGZPAD:
+ case FT_STRINGZTRUNC:
+ dissector_change_string(name, (!pattern)?"":(const gchar *) pattern, dissector);
+ return TRUE;
+ default:
+ return FALSE;
+ };
+
+ return TRUE;
+}
+
+/* Some useful utilities for Decode As */
+
+/*
+ * A list of dissectors that need to be reset.
+ */
+static GSList *dissector_reset_list = NULL;
+
+/*
+ * A callback function to parse each "decode as" entry in the file and apply the change
+ */
+static prefs_set_pref_e
+read_set_decode_as_entries(gchar *key, const gchar *value,
+ void *user_data,
+ gboolean return_range_errors _U_)
+{
+ gchar *values[4] = {NULL, NULL, NULL, NULL};
+ gchar delimiter[4] = {',', ',', ',','\0'};
+ gchar *pch;
+ guint i, j;
+ GHashTable* processed_entries = (GHashTable*)user_data;
+ dissector_table_t sub_dissectors;
+ prefs_set_pref_e retval = PREFS_SET_OK;
+ gboolean is_valid = FALSE;
+
+ if (strcmp(key, DECODE_AS_ENTRY) == 0) {
+ /* Parse csv into table, selector, initial, current */
+ for (i = 0; i < 4; i++) {
+ pch = strchr(value, delimiter[i]);
+ if (pch == NULL) {
+ for (j = 0; j < i; j++) {
+ g_free(values[j]);
+ }
+ return PREFS_SET_SYNTAX_ERR;
+ }
+ values[i] = g_strndup(value, pch - value);
+ value = pch + 1;
+ }
+ sub_dissectors = find_dissector_table(values[0]);
+ if (sub_dissectors != NULL) {
+ dissector_handle_t handle;
+ ftenum_t selector_type;
+ pref_t* pref_value;
+ module_t *module;
+ const char* proto_name;
+
+ selector_type = dissector_table_get_type(sub_dissectors);
+
+ handle = dissector_table_get_dissector_handle(sub_dissectors, values[3]);
+ if (handle != NULL || g_ascii_strcasecmp(values[3], DECODE_AS_NONE) == 0) {
+ is_valid = TRUE;
+ }
+
+ if (is_valid) {
+ if (FT_IS_STRING(selector_type)) {
+ dissector_change_string(values[0], values[1], handle);
+ } else {
+ char *p;
+ long long_value;
+
+ long_value = strtol(values[1], &p, 0);
+ if (p == values[0] || *p != '\0' || long_value < 0 ||
+ (unsigned long)long_value > UINT_MAX) {
+ retval = PREFS_SET_SYNTAX_ERR;
+ is_valid = FALSE;
+ } else {
+ dissector_change_uint(values[0], (guint)long_value, handle);
+ }
+
+ /* Now apply the value data back to dissector table preference */
+ if (handle != NULL) {
+ proto_name = proto_get_protocol_filter_name(dissector_handle_get_protocol_index(handle));
+ module = prefs_find_module(proto_name);
+ pref_value = prefs_find_preference(module, values[0]);
+ if (pref_value != NULL) {
+ gboolean replace = FALSE;
+ if (g_hash_table_lookup(processed_entries, proto_name) == NULL) {
+ /* First decode as entry for this protocol, ranges may be replaced */
+ replace = TRUE;
+
+ /* Remember we've processed this protocol */
+ g_hash_table_insert(processed_entries, (gpointer)proto_name, (gpointer)proto_name);
+ }
+
+ prefs_add_decode_as_value(pref_value, (guint)long_value, replace);
+ module->prefs_changed_flags |= prefs_get_effect_flags(pref_value);
+ }
+ }
+ }
+ }
+ if (is_valid) {
+ decode_build_reset_list(values[0], selector_type, values[1], NULL, NULL);
+ }
+ } else {
+ retval = PREFS_SET_SYNTAX_ERR;
+ }
+
+ } else {
+ retval = PREFS_SET_NO_SUCH_PREF;
+ }
+
+ for (i = 0; i < 4; i++) {
+ g_free(values[i]);
+ }
+ return retval;
+}
+
+void
+load_decode_as_entries(void)
+{
+ char *daf_path;
+ FILE *daf;
+
+ decode_clear_all();
+
+ daf_path = get_persconffile_path(DECODE_AS_ENTRIES_FILE_NAME, TRUE);
+ if ((daf = ws_fopen(daf_path, "r")) != NULL) {
+ /* Store saved entries for better range processing */
+ GHashTable* processed_entries = g_hash_table_new(g_str_hash, g_str_equal);
+ read_prefs_file(daf_path, daf, read_set_decode_as_entries, processed_entries);
+ g_hash_table_destroy(processed_entries);
+ fclose(daf);
+ }
+ g_free(daf_path);
+}
+
+
+/* Make a sorted list of the enties as we are fetching them from a hash table. Then write it out from the sorted list */
+static void
+decode_as_write_entry (const gchar *table_name, ftenum_t selector_type,
+ gpointer key, gpointer value, gpointer user_data)
+{
+ GList **decode_as_rows_list = (GList **)user_data;
+ dissector_handle_t current, initial;
+ const gchar *current_dissector_name, *initial_dissector_name, *decode_as_row;
+
+ current = dtbl_entry_get_handle((dtbl_entry_t *)value);
+ if (current == NULL)
+ current_dissector_name = DECODE_AS_NONE;
+ else
+ current_dissector_name = dissector_handle_get_description(current);
+ initial = dtbl_entry_get_initial_handle((dtbl_entry_t *)value);
+ if (initial == NULL)
+ initial_dissector_name = DECODE_AS_NONE;
+ else
+ initial_dissector_name = dissector_handle_get_description(initial);
+
+ switch (selector_type) {
+
+ case FT_UINT8:
+ case FT_UINT16:
+ case FT_UINT24:
+ case FT_UINT32:
+ /*
+ * XXX - write these in decimal, regardless of the base of
+ * the dissector table's selector, as older versions of
+ * Wireshark used atoi() when reading this file, and
+ * failed to handle hex or octal numbers.
+ *
+ * That will be fixed in future 1.10 and 1.12 releases,
+ * but pre-1.10 releases are at end-of-life and won't
+ * be fixed.
+ */
+ decode_as_row = ws_strdup_printf(
+ DECODE_AS_ENTRY ": %s,%u,%s,%s\n",
+ table_name, GPOINTER_TO_UINT(key), initial_dissector_name,
+ current_dissector_name);
+ break;
+ case FT_NONE:
+ /*
+ * XXX - Just put a placeholder for the key value. Currently
+ * FT_NONE dissector table uses a single uint value for
+ * a placeholder
+ */
+ decode_as_row = ws_strdup_printf(
+ DECODE_AS_ENTRY ": %s,0,%s,%s\n",
+ table_name, initial_dissector_name,
+ current_dissector_name);
+ break;
+
+ case FT_STRING:
+ case FT_STRINGZ:
+ case FT_UINT_STRING:
+ case FT_STRINGZPAD:
+ case FT_STRINGZTRUNC:
+ decode_as_row = ws_strdup_printf(
+ DECODE_AS_ENTRY ": %s,%s,%s,%s\n",
+ table_name, (gchar *)key, initial_dissector_name,
+ current_dissector_name);
+ break;
+
+ default:
+ ws_assert_not_reached();
+ break;
+ }
+
+ /* Do we need a better sort function ???*/
+ *decode_as_rows_list = g_list_insert_sorted (*decode_as_rows_list, (gpointer)decode_as_row,
+ (GCompareFunc)g_ascii_strcasecmp);
+
+}
+
+/* Print the sorted rows to File */
+static void
+decode_as_print_rows(gpointer data, gpointer user_data)
+{
+ FILE *da_file = (FILE *)user_data;
+ const gchar *decode_as_row = (const gchar *)data;
+
+ fprintf(da_file, "%s",decode_as_row);
+
+}
+int
+save_decode_as_entries(gchar** err)
+{
+ char *pf_dir_path;
+ char *daf_path;
+ FILE *da_file;
+ GList *decode_as_rows_list = NULL;
+
+ if (create_persconffile_dir(&pf_dir_path) == -1) {
+ *err = ws_strdup_printf("Can't create directory\n\"%s\"\nfor recent file: %s.",
+ pf_dir_path, g_strerror(errno));
+ g_free(pf_dir_path);
+ return -1;
+ }
+
+ daf_path = get_persconffile_path(DECODE_AS_ENTRIES_FILE_NAME, TRUE);
+ if ((da_file = ws_fopen(daf_path, "w")) == NULL) {
+ *err = ws_strdup_printf("Can't open decode_as_entries file\n\"%s\": %s.",
+ daf_path, g_strerror(errno));
+ g_free(daf_path);
+ return -1;
+ }
+
+ fprintf(da_file, "# \"Decode As\" entries file for %s " VERSION ".\n"
+ "#\n"
+ "# This file is regenerated each time \"Decode As\" preferences\n"
+ "# are saved within %s. Making manual changes should be safe,\n"
+ "# however.\n",
+ get_configuration_namespace(), get_configuration_namespace());
+
+ dissector_all_tables_foreach_changed(decode_as_write_entry, &decode_as_rows_list);
+
+ g_list_foreach(decode_as_rows_list, decode_as_print_rows, da_file);
+
+ fclose(da_file);
+ g_free(daf_path);
+ g_list_free_full(decode_as_rows_list, g_free);
+
+ return 0;
+}
+
+/*
+ * Data structure for tracking which dissector need to be reset. This
+ * structure is necessary as a hash table entry cannot be removed
+ * while a g_hash_table_foreach walk is in progress.
+ */
+typedef struct dissector_delete_item {
+ /* The name of the dissector table */
+ gchar *ddi_table_name;
+ /* The type of the selector in that dissector table */
+ ftenum_t ddi_selector_type;
+ /* The selector in the dissector table */
+ union {
+ guint sel_uint;
+ char *sel_string;
+ } ddi_selector;
+} dissector_delete_item_t;
+
+void
+decode_build_reset_list (const gchar *table_name, ftenum_t selector_type,
+ gpointer key, gpointer value _U_,
+ gpointer user_data _U_)
+{
+ dissector_delete_item_t *item;
+
+ item = g_new(dissector_delete_item_t,1);
+ item->ddi_table_name = g_strdup(table_name);
+ item->ddi_selector_type = selector_type;
+ switch (selector_type) {
+
+ case FT_UINT8:
+ case FT_UINT16:
+ case FT_UINT24:
+ case FT_UINT32:
+ item->ddi_selector.sel_uint = GPOINTER_TO_UINT(key);
+ break;
+
+ case FT_NONE:
+ /* Not really needed, but prevents the assert */
+ item->ddi_selector.sel_uint = 0;
+ break;
+
+ case FT_STRING:
+ case FT_STRINGZ:
+ case FT_UINT_STRING:
+ case FT_STRINGZPAD:
+ case FT_STRINGZTRUNC:
+ item->ddi_selector.sel_string = g_strdup((char *)key);
+ break;
+
+ default:
+ ws_assert_not_reached();
+ }
+ dissector_reset_list = g_slist_prepend(dissector_reset_list, item);
+}
+
+/* clear all settings */
+void
+decode_clear_all(void)
+{
+ dissector_delete_item_t *item;
+ GSList *tmp;
+
+ dissector_all_tables_foreach_changed(decode_build_reset_list, NULL);
+
+ for (tmp = dissector_reset_list; tmp; tmp = g_slist_next(tmp)) {
+ item = (dissector_delete_item_t *)tmp->data;
+ switch (item->ddi_selector_type) {
+
+ case FT_UINT8:
+ case FT_UINT16:
+ case FT_UINT24:
+ case FT_UINT32:
+ dissector_reset_uint(item->ddi_table_name,
+ item->ddi_selector.sel_uint);
+ break;
+
+ case FT_NONE:
+ dissector_reset_payload(item->ddi_table_name);
+ break;
+
+ case FT_STRING:
+ case FT_STRINGZ:
+ case FT_UINT_STRING:
+ case FT_STRINGZPAD:
+ case FT_STRINGZTRUNC:
+ dissector_reset_string(item->ddi_table_name,
+ item->ddi_selector.sel_string);
+ g_free(item->ddi_selector.sel_string);
+ break;
+
+ default:
+ ws_assert_not_reached();
+ }
+ g_free(item->ddi_table_name);
+ g_free(item);
+ }
+ g_slist_free(dissector_reset_list);
+ dissector_reset_list = NULL;
+
+ decode_dcerpc_reset_all();
+}
+
+void
+decode_cleanup(void)
+{
+ g_list_free(decode_as_list);
+ decode_as_list = NULL;
+}
+
+/*
+ * Editor modelines
+ *
+ * Local Variables:
+ * c-basic-offset: 4
+ * tab-width: 8
+ * indent-tabs-mode: nil
+ * End:
+ *
+ * ex: set shiftwidth=4 tabstop=8 expandtab:
+ * :indentSize=4:tabSize=8:noTabs=true:
+ */