summaryrefslogtreecommitdiff
path: root/gtk/proto_dlg.c
diff options
context:
space:
mode:
authorLaurent Deniel <laurent.deniel@free.fr>2000-08-13 14:09:15 +0000
committerLaurent Deniel <laurent.deniel@free.fr>2000-08-13 14:09:15 +0000
commitcc36f0b9312fc36778bf227cbaaf02c89baf09b3 (patch)
tree0aff9b70a00a628a713030dc9000f3b5492d492b /gtk/proto_dlg.c
parentdb31ba13c8b9f82c1b96b5444fe3b7e8ff109ce1 (diff)
downloadwireshark-cc36f0b9312fc36778bf227cbaaf02c89baf09b3.tar.gz
Add the "Edit:Protocols..." feature which currently only implements
the following: It is now possible to enable/disable a particular protocol decoding (i.e. the protocol dissector is void or not). When a protocol is disabled, it is displayed as Data and of course, all linked sub-protocols are disabled as well. Disabling a protocol could be interesting: - in case of buggy dissectors - in case of wrong heuristics - for performance reasons - to decode the data as another protocol (TODO) Currently (if I am not wrong), all dissectors but NFS can be disabled (and dissectors that do not register protocols :-) I do not like the way the RPC sub-dissectors are disabled (in the sub-dissectors) since this could be done in the RPC dissector itself, knowing the sub-protocol hfinfo entry (this is why, I've not modified the NFS one yet). Two functions are added in proto.c : gboolean proto_is_protocol_enabled(int n); void proto_set_decoding(int n, gboolean enabled); and two MACROs which can be used in dissectors: OLD_CHECK_DISPLAY_AS_DATA(index, pd, offset, fd, tree) CHECK_DISPLAY_AS_DATA(index, tvb, pinfo, tree) See also the XXX in proto_dlg.c and proto.c around the new functions. svn path=/trunk/; revision=2267
Diffstat (limited to 'gtk/proto_dlg.c')
-rw-r--r--gtk/proto_dlg.c287
1 files changed, 287 insertions, 0 deletions
diff --git a/gtk/proto_dlg.c b/gtk/proto_dlg.c
new file mode 100644
index 0000000000..2e38043bf5
--- /dev/null
+++ b/gtk/proto_dlg.c
@@ -0,0 +1,287 @@
+/* proto_dlg.c
+ *
+ * $Id: proto_dlg.c,v 1.1 2000/08/13 14:03:49 deniel Exp $
+ *
+ * Laurent Deniel <deniel@worldnet.fr>
+ *
+ * Ethereal - Network traffic analyzer
+ * By Gerald Combs <gerald@zing.org>
+ * Copyright 2000 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.
+ *
+ */
+
+/*
+ * TODO :
+ *
+ * Modify proto.c to have a better protocol characteristics database
+ * such as ordered list or hash table. This would allow a quick search
+ * knowing the protocol abbreviation and to enhance this stuff by adding
+ * other fields (hfinfo is currently limited since protocols and fields
+ * share the same structure type).
+ *
+ * The protocol display should be sorted (by abbreviation).
+ *
+ */
+
+#ifdef HAVE_CONFIG_H
+# include "config.h"
+#endif
+
+#include <gtk/gtk.h>
+#include <stdlib.h>
+#include <stdio.h>
+#include <string.h>
+
+#ifdef HAVE_SYS_TYPES_H
+#include <sys/types.h>
+#endif
+
+#include "prefs.h"
+#include "globals.h"
+#include "gtkglobals.h"
+#include "main.h"
+#include "util.h"
+#include "ui_util.h"
+#include "dlg_utils.h"
+#include "proto_dlg.h"
+
+static void proto_ok_cb(GtkWidget *, gpointer);
+static void proto_apply_cb(GtkWidget *, gpointer);
+static void proto_close_cb(GtkWidget *, gpointer);
+
+static void show_proto_selection(GtkWidget *main, GtkWidget *container);
+static gboolean set_proto_selection(GtkWidget *);
+
+static GtkWidget *proto_w = NULL;
+
+void proto_cb(GtkWidget *w, gpointer data)
+{
+
+ GtkWidget *main_vb, *bbox, *proto_nb, *apply_bt, *cancel_bt, *ok_bt,
+ *label, *scrolled_w, *selection_vb;
+
+ if (proto_w != NULL) {
+ reactivate_window(proto_w);
+ return;
+ }
+
+ proto_w = dlg_window_new();
+ gtk_window_set_title(GTK_WINDOW(proto_w), "Ethereal: Protocol");
+ gtk_signal_connect(GTK_OBJECT(proto_w), "destroy",
+ GTK_SIGNAL_FUNC(proto_close_cb), NULL);
+ gtk_widget_set_usize(GTK_WIDGET(proto_w), DEF_WIDTH * 2/3, DEF_HEIGHT * 2/3);
+
+ /* Container for each row of widgets */
+
+ main_vb = gtk_vbox_new(FALSE, 0);
+ gtk_container_border_width(GTK_CONTAINER(main_vb), 1);
+ gtk_container_add(GTK_CONTAINER(proto_w), main_vb);
+ gtk_widget_show(main_vb);
+
+ /* Protocol topics container */
+
+ proto_nb = gtk_notebook_new();
+ gtk_container_add(GTK_CONTAINER(main_vb), proto_nb);
+ /* XXX do not know why I need this to fill all space around buttons */
+ gtk_widget_set_usize(GTK_WIDGET(proto_nb), DEF_WIDTH * 2/3 - 50,
+ DEF_HEIGHT * 2/3 - 50);
+
+ /* Protocol selection panel ("enable/disable" protocols) */
+
+ selection_vb = gtk_vbox_new(FALSE, 0);
+ gtk_container_border_width(GTK_CONTAINER(selection_vb), 1);
+ label = gtk_label_new("Button pressed: protocol decoding is enabled");
+ gtk_widget_show(label);
+ gtk_box_pack_start(GTK_BOX(selection_vb), label, FALSE, FALSE, 0);
+ scrolled_w = gtk_scrolled_window_new(NULL, NULL);
+ gtk_container_set_border_width(GTK_CONTAINER(scrolled_w), 1);
+ gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled_w),
+ GTK_POLICY_AUTOMATIC,
+ GTK_POLICY_ALWAYS);
+ set_scrollbar_placement_scrollw(scrolled_w, prefs.gui_scrollbar_on_right);
+ remember_scrolled_window(scrolled_w);
+ gtk_box_pack_start(GTK_BOX(selection_vb), scrolled_w, TRUE, TRUE, 0);
+ show_proto_selection(proto_w, scrolled_w);
+ gtk_widget_show(scrolled_w);
+ gtk_widget_show(selection_vb);
+ label = gtk_label_new("Decoding");
+ gtk_notebook_append_page(GTK_NOTEBOOK(proto_nb), selection_vb, label);
+ label = gtk_label_new("Note that when a protocol is disabled, "
+ "all linked sub-protocols are as well");
+ gtk_widget_show(label);
+ gtk_box_pack_start(GTK_BOX(selection_vb), label, FALSE, FALSE, 0);
+
+ /* XXX add other protocol-related panels here ... */
+
+ gtk_widget_show(proto_nb);
+
+ /* Ok, Apply, Cancel Buttons */
+
+ bbox = gtk_hbutton_box_new();
+ gtk_button_box_set_layout(GTK_BUTTON_BOX(bbox), GTK_BUTTONBOX_END);
+ gtk_button_box_set_spacing(GTK_BUTTON_BOX(bbox), 5);
+ gtk_container_add(GTK_CONTAINER(main_vb), bbox);
+ gtk_widget_show(bbox);
+
+ ok_bt = gtk_button_new_with_label ("OK");
+ gtk_signal_connect(GTK_OBJECT(ok_bt), "clicked",
+ GTK_SIGNAL_FUNC(proto_ok_cb), GTK_OBJECT(proto_w));
+ GTK_WIDGET_SET_FLAGS(ok_bt, GTK_CAN_DEFAULT);
+ gtk_box_pack_start(GTK_BOX (bbox), ok_bt, TRUE, TRUE, 0);
+ gtk_widget_grab_default(ok_bt);
+ gtk_widget_show(ok_bt);
+
+ apply_bt = gtk_button_new_with_label ("Apply");
+ gtk_signal_connect(GTK_OBJECT(apply_bt), "clicked",
+ GTK_SIGNAL_FUNC(proto_apply_cb), GTK_OBJECT(proto_w));
+ GTK_WIDGET_SET_FLAGS(apply_bt, GTK_CAN_DEFAULT);
+ gtk_box_pack_start(GTK_BOX (bbox), apply_bt, TRUE, TRUE, 0);
+ gtk_widget_show(apply_bt);
+
+ cancel_bt = gtk_button_new_with_label ("Cancel");
+ gtk_signal_connect(GTK_OBJECT(cancel_bt), "clicked",
+ GTK_SIGNAL_FUNC(proto_close_cb), GTK_OBJECT(proto_w));
+ GTK_WIDGET_SET_FLAGS(cancel_bt, GTK_CAN_DEFAULT);
+ gtk_box_pack_start(GTK_BOX (bbox), cancel_bt, TRUE, TRUE, 0);
+ gtk_widget_show(cancel_bt);
+
+ dlg_set_cancel(proto_w, cancel_bt);
+
+ gtk_quit_add_destroy(gtk_main_level(), GTK_OBJECT(proto_w));
+ gtk_widget_show(proto_w);
+
+} /* proto_cb */
+
+static void proto_close_cb(GtkWidget *w, gpointer data)
+{
+ if (proto_w)
+ gtk_widget_destroy(proto_w);
+ proto_w = NULL;
+}
+
+static void proto_ok_cb(GtkWidget *ok_bt, gpointer parent_w)
+{
+ gboolean redissect;
+ redissect = set_proto_selection(GTK_WIDGET(parent_w));
+ gtk_widget_destroy(GTK_WIDGET(parent_w));
+ if (redissect)
+ redissect_packets(&cfile);
+}
+
+static void proto_apply_cb(GtkWidget *ok_bt, gpointer parent_w)
+{
+ if (set_proto_selection(GTK_WIDGET(parent_w)))
+ redissect_packets(&cfile);
+}
+
+static gboolean set_proto_selection(GtkWidget *parent_w)
+{
+ int i;
+ gboolean need_redissect = FALSE;
+
+ for (i = 0; i < proto_registrar_n() ; i++) {
+ if (proto_registrar_is_protocol(i)) {
+ GtkWidget *button;
+ header_field_info *hfinfo;
+ hfinfo = proto_registrar_get_nth(i);
+
+ if (strcmp(hfinfo->abbrev, "data") == 0 ||
+ strcmp(hfinfo->abbrev, "text") == 0 ||
+ strcmp(hfinfo->abbrev, "malformed") == 0 ||
+ strcmp(hfinfo->abbrev, "short") == 0 ||
+ strcmp(hfinfo->abbrev, "frame") == 0) continue;
+
+ button = (GtkWidget *) gtk_object_get_data(GTK_OBJECT(parent_w),
+ hfinfo->abbrev);
+ /* XXX optimization but should not use display field */
+ if (hfinfo->display != GTK_TOGGLE_BUTTON (button)->active) {
+ proto_set_decoding(i, GTK_TOGGLE_BUTTON (button)->active);
+ need_redissect = TRUE;
+ }
+ }
+ }
+
+ return need_redissect;
+
+} /* set_proto_selection */
+
+static void show_proto_selection(GtkWidget *main, GtkWidget *container)
+{
+
+ /* XXX : we should sort the protocol abbrev */
+
+#define NB_COL 7
+
+ GtkTooltips *tooltips;
+ GtkWidget *table;
+ int i, t = 0, l = 0, nb_line, nb_proto = 0;
+
+ /* Obtain the number of "true" protocols */
+
+ for (i = 0; i < proto_registrar_n() ; i++) {
+ if (proto_registrar_is_protocol(i)) {
+ nb_proto ++;
+ }
+ }
+
+ /* XXX ignore "data", "malformed", "short", "frame", "text" */
+ nb_proto -= 5;
+
+ /* create a table (n x NB_COL) of buttons */
+
+ nb_line = (nb_proto % NB_COL) ? nb_proto / NB_COL + 1 : nb_proto / NB_COL;
+ table = gtk_table_new (nb_line, NB_COL, FALSE);
+ gtk_table_set_row_spacings(GTK_TABLE (table), 1);
+ gtk_table_set_col_spacings(GTK_TABLE (table), 1);
+ gtk_scrolled_window_add_with_viewport(GTK_SCROLLED_WINDOW(container), table);
+ gtk_widget_show(table);
+
+ tooltips = gtk_tooltips_new();
+
+ nb_proto = 0;
+
+ for (i = 0; i < proto_registrar_n() ; i++) {
+ if (proto_registrar_is_protocol(i)) {
+ GtkWidget *button;
+ header_field_info *hfinfo;
+ hfinfo = proto_registrar_get_nth(i);
+
+ if (strcmp(hfinfo->abbrev, "data") == 0 ||
+ strcmp(hfinfo->abbrev, "text") == 0 ||
+ strcmp(hfinfo->abbrev, "malformed") == 0 ||
+ strcmp(hfinfo->abbrev, "short") == 0 ||
+ strcmp(hfinfo->abbrev, "frame") == 0) continue;
+
+ /* button label is the protocol abbrev */
+ button = gtk_toggle_button_new_with_label (hfinfo->abbrev);
+ /* tip is the complete protocol name */
+ gtk_tooltips_set_tip(tooltips, button, hfinfo->name, NULL);
+ gtk_toggle_button_set_state(GTK_TOGGLE_BUTTON(button), hfinfo->display);
+ gtk_object_set_data(GTK_OBJECT(main), hfinfo->abbrev, button);
+ gtk_table_attach_defaults (GTK_TABLE (table), button, l, l+1, t, t+1);
+ gtk_widget_show (button);
+ if (++nb_proto % NB_COL) {
+ l++;
+ }
+ else {
+ l = 0;
+ t++;
+ }
+ }
+ }
+
+} /* show_proto_selection */