summaryrefslogtreecommitdiffstats
path: root/epan/funnel.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/funnel.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/funnel.c')
-rw-r--r--epan/funnel.c354
1 files changed, 354 insertions, 0 deletions
diff --git a/epan/funnel.c b/epan/funnel.c
new file mode 100644
index 00000000..276559d7
--- /dev/null
+++ b/epan/funnel.c
@@ -0,0 +1,354 @@
+/*
+ * funnel.c
+ *
+ * EPAN's GUI mini-API
+ *
+ * (c) 2006, Luis E. Garcia Ontanon <luis@ontanon.org>
+ *
+ * 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/funnel.h>
+#include <wsutil/glib-compat.h>
+
+typedef struct _funnel_menu_t {
+ char *name;
+ register_stat_group_t group;
+ funnel_menu_callback callback;
+ gpointer callback_data;
+ funnel_menu_callback_data_free callback_data_free;
+ gboolean retap;
+ struct _funnel_menu_t* next;
+} funnel_menu_t;
+
+typedef struct _console_menu {
+ char *name;
+ funnel_console_eval_cb_t eval_cb;
+ funnel_console_open_cb_t open_cb;
+ funnel_console_close_cb_t close_cb;
+ void *user_data;
+ funnel_console_data_free_cb_t data_free_cb;
+} funnel_console_menu_t;
+
+/* XXX This assumes one main window and one capture file. */
+static const funnel_ops_t* ops = NULL;
+static funnel_menu_t* registered_menus = NULL;
+static funnel_menu_t* added_menus = NULL;
+static funnel_menu_t* removed_menus = NULL;
+static gboolean menus_registered = FALSE;
+
+/**
+ * Represents a single packet menu entry and callback
+ */
+typedef struct _funnel_packet_menu_t {
+ char *name; /**< Name to display in the GUI */
+ char *required_fields; /**< comma-separated list of fields
+ that must be present for the
+ packet menu to be displayed */
+ funnel_packet_menu_callback callback; /**< Lua function to be called on
+ menu item selection. */
+ gpointer callback_data; /**< Lua state for the callback
+ function */
+ gboolean retap; /**< Whether or not to rescan the
+ capture file's packets */
+ struct _funnel_packet_menu_t* next; /**< Pointer to the next
+ _funnel_packet_menu_t for the
+ singly-linked list
+ implemenation */
+} funnel_packet_menu_t;
+
+/*
+ * List of all registered funnel_packet_menu_t's
+ */
+static funnel_packet_menu_t* registered_packet_menus = NULL;
+
+static GSList *registered_console_menus = NULL;
+
+/*
+ * TRUE if the packet menus were modified since the last registration
+ */
+static gboolean packet_menus_modified = FALSE;
+static void funnel_clear_packet_menu (funnel_packet_menu_t** menu_list);
+
+const funnel_ops_t* funnel_get_funnel_ops(void) { return ops; }
+void funnel_set_funnel_ops(const funnel_ops_t* o) { ops = o; }
+
+static void funnel_clear_console_menu(void);
+
+static void funnel_insert_menu (funnel_menu_t** menu_list, funnel_menu_t *menu)
+{
+ if (!(*menu_list)) {
+ *menu_list = menu;
+ } else {
+ funnel_menu_t* c;
+ for (c = *menu_list; c->next; c = c->next);
+ c->next = menu;
+ }
+}
+
+static void funnel_remove_menu (funnel_menu_t ** menu_list, funnel_menu_t *menu)
+{
+ funnel_menu_t *m = *menu_list, *p = NULL;
+
+ while (m) {
+ if (m->callback == menu->callback) {
+ if (p) {
+ p->next = m->next;
+ } else {
+ *menu_list = m->next;
+ }
+ g_free(m->name);
+ if (m->callback_data_free) {
+ m->callback_data_free(m->callback_data);
+ }
+ g_free(m);
+ if (p) {
+ m = p->next;
+ } else {
+ m = *menu_list;
+ }
+ } else {
+ p = m;
+ m = m->next;
+ }
+ }
+}
+
+static void funnel_clear_menu (funnel_menu_t** menu_list)
+{
+ funnel_menu_t *m;
+
+ while (*menu_list) {
+ m = *menu_list;
+ *menu_list = m->next;
+ g_free(m->name);
+ g_free(m);
+ }
+ *menu_list = NULL;
+}
+
+void funnel_register_menu(const char *name,
+ register_stat_group_t group,
+ funnel_menu_callback callback,
+ gpointer callback_data,
+ funnel_menu_callback_data_free callback_data_free,
+ gboolean retap)
+{
+ funnel_menu_t* m = g_new(funnel_menu_t, 1);
+ m->name = g_strdup(name);
+ m->group = group;
+ m->callback = callback;
+ m->callback_data = callback_data;
+ m->callback_data_free = callback_data_free;
+ m->retap = retap;
+ m->next = NULL;
+
+ funnel_insert_menu(&registered_menus, m);
+ if (menus_registered) {
+ funnel_menu_t* m_r = (funnel_menu_t *)g_memdup2(m, sizeof *m);
+ m_r->name = g_strdup(name);
+ funnel_insert_menu(&added_menus, m_r);
+ }
+}
+
+void funnel_deregister_menus(funnel_menu_callback callback)
+{
+ funnel_menu_t* m = g_new0(funnel_menu_t, 1);
+ m->callback = callback;
+
+ funnel_remove_menu(&registered_menus, m);
+ funnel_insert_menu(&removed_menus, m);
+
+ // Clear and free memory of packet menus
+ funnel_clear_packet_menu(&registered_packet_menus);
+ packet_menus_modified = TRUE;
+}
+
+void funnel_register_all_menus(funnel_registration_cb_t r_cb)
+{
+ funnel_menu_t* c;
+ for (c = registered_menus; c; c = c->next) {
+ r_cb(c->name,c->group,c->callback,c->callback_data,c->retap);
+ }
+ menus_registered = TRUE;
+}
+
+void funnel_reload_menus(funnel_deregistration_cb_t d_cb,
+ funnel_registration_cb_t r_cb)
+{
+ funnel_menu_t* c;
+ for (c = removed_menus; c; c = c->next) {
+ d_cb(c->callback);
+ }
+ for (c = added_menus; c; c = c->next) {
+ r_cb(c->name,c->group,c->callback,c->callback_data,c->retap);
+ }
+
+ funnel_clear_menu(&removed_menus);
+ funnel_clear_menu(&added_menus);
+}
+
+
+/*
+ * Inserts a funnel_packet_menu_t into a list of funnel_packet_menu_t's
+ *
+ * @param menu_list the list of menus that the menu will be added to
+ * @param menu the menu to add to the list of menus
+ */
+static void funnel_insert_packet_menu (funnel_packet_menu_t** menu_list, funnel_packet_menu_t *menu)
+{
+ if (!(*menu_list)) {
+ *menu_list = menu;
+ } else {
+ funnel_packet_menu_t* c;
+ for (c = *menu_list; c->next; c = c->next);
+ c->next = menu;
+ }
+}
+
+/**
+ * Entry point for Lua code to register a packet menu
+ *
+ * Stores the menu name and callback from the Lua code
+ * into registered_packet_menus so that the
+ * Wireshark GUI code can retrieve it with
+ * funnel_register_all_packet_menus().
+ */
+void funnel_register_packet_menu(const char *name,
+ const char *required_fields,
+ funnel_packet_menu_callback callback,
+ gpointer callback_data,
+ gboolean retap)
+{
+ funnel_packet_menu_t* m = g_new0(funnel_packet_menu_t, 1);
+ m->name = g_strdup(name);
+ m->required_fields = g_strdup(required_fields);
+ m->callback = callback;
+ m->callback_data = callback_data;
+ m->retap = retap;
+ m->next = NULL;
+
+ funnel_insert_packet_menu(&registered_packet_menus, m);
+ packet_menus_modified = TRUE;
+}
+
+/**
+ * Clears a list of funnel_packet_menu_t's and free()s all associated memory
+ *
+ * @param menu_list the list of menus to clear
+ */
+static void funnel_clear_packet_menu (funnel_packet_menu_t** menu_list)
+{
+ funnel_packet_menu_t *m;
+
+ while (*menu_list) {
+ m = *menu_list;
+ *menu_list = m->next;
+ g_free(m->name);
+ g_free(m->required_fields);
+ if (m->callback_data) {
+ g_free(m->callback_data);
+ }
+ g_free(m);
+ }
+ *menu_list = NULL;
+}
+
+/**
+ * Entry point for Wireshark GUI to obtain all registered packet menus
+ *
+ * Calls the supplied callback for each packet menu registered with
+ * funnel_register_packet_menu().
+ *
+ * @param r_cb the callback function to call with each registered packet menu
+ */
+void funnel_register_all_packet_menus(funnel_registration_packet_cb_t r_cb)
+{
+ funnel_packet_menu_t* c;
+ for (c = registered_packet_menus; c; c = c->next) {
+ r_cb(c->name,c->required_fields,c->callback,c->callback_data,c->retap);
+ }
+ packet_menus_modified = FALSE;
+}
+
+/**
+ * Returns whether the packet menus have been modified since they were last registered
+ *
+ * @return TRUE if the packet menus were modified since the last registration
+ */
+gboolean funnel_packet_menus_modified(void)
+{
+ return packet_menus_modified;
+}
+
+void funnel_cleanup(void)
+{
+ funnel_clear_menu(&registered_menus);
+ funnel_clear_packet_menu(&registered_packet_menus);
+ funnel_clear_console_menu();
+}
+
+/**
+ * Entry point for code to register a console menu
+ */
+void funnel_register_console_menu(const char *name,
+ funnel_console_eval_cb_t eval_cb,
+ funnel_console_open_cb_t open_cb,
+ funnel_console_close_cb_t close_cb,
+ gpointer callback_data,
+ funnel_console_data_free_cb_t free_data)
+{
+ funnel_console_menu_t* m = g_new0(funnel_console_menu_t, 1);
+ m->name = g_strdup(name);
+ m->eval_cb = eval_cb;
+ m->open_cb = open_cb;
+ m->close_cb = close_cb;
+ m->user_data = callback_data;
+ m->data_free_cb = free_data;
+
+ registered_console_menus = g_slist_prepend(registered_console_menus, m);
+}
+
+void funnel_register_all_console_menus(funnel_registration_console_cb_t r_cb)
+{
+ GSList *l;
+ for (l = registered_console_menus; l != NULL; l = l->next) {
+ funnel_console_menu_t *m = l->data;
+ r_cb(m->name, m->eval_cb, m->open_cb, m->close_cb, m->user_data);
+ }
+}
+
+static void funnel_clear_console_menu(void)
+{
+ GSList *l;
+ for (l = registered_console_menus; l != NULL; l = l->next) {
+ funnel_console_menu_t *m = l->data;
+ g_free(m->name);
+ if (m->data_free_cb && m->user_data) {
+ m->data_free_cb(m->user_data);
+ }
+ g_free(l->data);
+ l->data = NULL;
+ }
+ g_slist_free(registered_console_menus);
+ registered_console_menus = NULL;
+}
+
+/*
+ * 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:
+ */