forked from osmocom/wireshark
b4471e0195
set the read filter dialog modal and transient to the parent window if requested. This way, it will receive input signals (solving problems with GTK2's gtk_file_chooser). To do this, add another construct_args flag, so it will be modal only if really needed ... svn path=/trunk/; revision=16926
814 lines
28 KiB
C
814 lines
28 KiB
C
/* find_dlg.c
|
|
* Routines for "find frame" window
|
|
*
|
|
* $Id$
|
|
*
|
|
* Ethereal - Network traffic analyzer
|
|
* By Gerald Combs <gerald@ethereal.com>
|
|
* Copyright 1998 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.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#include <string.h>
|
|
#include <ctype.h>
|
|
|
|
#include <gtk/gtk.h>
|
|
|
|
#include <epan/proto.h>
|
|
#include <epan/dfilter/dfilter.h>
|
|
#include <epan/strutil.h>
|
|
#include "globals.h"
|
|
|
|
#include "gui_utils.h"
|
|
#include "find_dlg.h"
|
|
#include "filter_dlg.h"
|
|
#include "alert_box.h"
|
|
#include "simple_dialog.h"
|
|
#include "dlg_utils.h"
|
|
#include "compat_macros.h"
|
|
#include <epan/prefs.h>
|
|
#include "prefs_dlg.h"
|
|
#include "keys.h"
|
|
#include "help_dlg.h"
|
|
|
|
/* Capture callback data keys */
|
|
#define E_FIND_FILT_KEY "find_filter_te"
|
|
#define E_FIND_BACKWARD_KEY "find_backward"
|
|
#define E_FIND_HEXDATA_KEY "find_hex"
|
|
#define E_FIND_STRINGDATA_KEY "find_string"
|
|
#define E_FIND_FILTERDATA_KEY "find_filter"
|
|
#define E_FIND_STRINGTYPE_KEY "find_string_type"
|
|
#define E_FIND_STRINGTYPE_LABEL_KEY "find_string_type_label"
|
|
#define E_CASE_SEARCH_KEY "case_insensitive_search"
|
|
#define E_SOURCE_HEX_KEY "hex_data_source"
|
|
#define E_SOURCE_DECODE_KEY "decode_data_source"
|
|
#define E_SOURCE_SUMMARY_KEY "summary_data_source"
|
|
#define E_FILT_TE_BUTTON_KEY "find_filter_button"
|
|
|
|
static gboolean case_type = TRUE;
|
|
static gboolean summary_data = FALSE;
|
|
static gboolean decode_data = FALSE;
|
|
|
|
static void
|
|
find_filter_te_syntax_check_cb(GtkWidget *w, gpointer parent_w);
|
|
|
|
static void
|
|
find_frame_ok_cb(GtkWidget *ok_bt, gpointer parent_w);
|
|
|
|
static void
|
|
find_frame_close_cb(GtkWidget *close_bt, gpointer parent_w);
|
|
|
|
static void
|
|
find_frame_destroy_cb(GtkWidget *win, gpointer user_data);
|
|
|
|
static void
|
|
hex_selected_cb(GtkWidget *button_rb _U_, gpointer parent_w);
|
|
|
|
static void
|
|
string_selected_cb(GtkWidget *button_rb _U_, gpointer parent_w);
|
|
|
|
static void
|
|
filter_selected_cb(GtkWidget *button_rb _U_, gpointer parent_w);
|
|
|
|
/*
|
|
* Keep a static pointer to the current "Find Packet" window, if any, so
|
|
* that if somebody tries to do "Find Packet" while there's already a
|
|
* "Find Packet" window up, we just pop up the existing one, rather than
|
|
* creating a new one.
|
|
*/
|
|
static GtkWidget *find_frame_w;
|
|
static GtkWidget *filter_text_box;
|
|
|
|
void
|
|
find_frame_cb(GtkWidget *w _U_, gpointer d _U_)
|
|
{
|
|
GtkWidget *main_vb, *main_find_hb, *main_options_hb,
|
|
|
|
*find_type_frame, *find_type_vb,
|
|
*find_type_hb, *find_type_lb, *hex_rb, *string_rb, *filter_rb,
|
|
*filter_hb, *filter_bt,
|
|
|
|
*direction_frame, *direction_vb,
|
|
*up_rb, *down_rb,
|
|
|
|
*data_frame, *data_vb,
|
|
*hex_data_rb, *decode_data_rb, *summary_data_rb,
|
|
|
|
*string_opt_frame, *string_opt_vb,
|
|
*case_cb, *combo_lb, *combo_cb,
|
|
|
|
*bbox, *ok_bt, *cancel_bt, *help_bt;
|
|
GtkTooltips *tooltips;
|
|
#if GTK_MAJOR_VERSION < 2
|
|
GtkAccelGroup *accel_group;
|
|
#endif
|
|
GList *glist = NULL;
|
|
/* No Apply button, but "OK" not only sets our text widget, it
|
|
activates it (i.e., it causes us to do the search). */
|
|
static construct_args_t args = {
|
|
"Ethereal: Search Filter",
|
|
FALSE,
|
|
TRUE,
|
|
FALSE
|
|
};
|
|
|
|
if (find_frame_w != NULL) {
|
|
/* There's already a "Find Packet" dialog box; reactivate it. */
|
|
reactivate_window(find_frame_w);
|
|
return;
|
|
}
|
|
|
|
find_frame_w = dlg_window_new("Ethereal: Find Packet");
|
|
|
|
tooltips = gtk_tooltips_new ();
|
|
|
|
#if GTK_MAJOR_VERSION < 2
|
|
/* Accelerator group for the accelerators (or, as they're called in
|
|
Windows and, I think, in Motif, "mnemonics"; Alt+<key> is a mnemonic,
|
|
Ctrl+<key> is an accelerator). */
|
|
accel_group = gtk_accel_group_new();
|
|
gtk_window_add_accel_group(GTK_WINDOW(find_frame_w), accel_group);
|
|
#endif
|
|
|
|
/* Container for each row of widgets */
|
|
main_vb = gtk_vbox_new(FALSE, 3);
|
|
gtk_container_border_width(GTK_CONTAINER(main_vb), 5);
|
|
gtk_container_add(GTK_CONTAINER(find_frame_w), main_vb);
|
|
gtk_widget_show(main_vb);
|
|
|
|
|
|
/* */
|
|
main_find_hb = gtk_hbox_new(FALSE, 3);
|
|
gtk_container_add(GTK_CONTAINER(main_vb), main_find_hb);
|
|
gtk_widget_show(main_find_hb);
|
|
|
|
|
|
/* find frame */
|
|
find_type_frame = gtk_frame_new("Find");
|
|
gtk_box_pack_start(GTK_BOX(main_find_hb), find_type_frame, TRUE, TRUE, 0);
|
|
gtk_widget_show(find_type_frame);
|
|
|
|
find_type_vb = gtk_vbox_new(FALSE, 3);
|
|
gtk_container_border_width(GTK_CONTAINER(find_type_vb), 3);
|
|
gtk_container_add(GTK_CONTAINER(find_type_frame), find_type_vb);
|
|
gtk_widget_show(find_type_vb);
|
|
|
|
/* find type row */
|
|
find_type_hb = gtk_hbox_new(FALSE, 3);
|
|
gtk_container_add(GTK_CONTAINER(find_type_vb), find_type_hb);
|
|
gtk_widget_show(find_type_hb);
|
|
|
|
find_type_lb = gtk_label_new("By:");
|
|
gtk_box_pack_start(GTK_BOX(find_type_hb), find_type_lb, FALSE, FALSE, 0);
|
|
gtk_widget_show(find_type_lb);
|
|
|
|
/* Filter */
|
|
filter_rb = RADIO_BUTTON_NEW_WITH_MNEMONIC(NULL,
|
|
"_Display filter", accel_group);
|
|
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(filter_rb), !cfile.hex && !cfile.string);
|
|
gtk_box_pack_start(GTK_BOX(find_type_hb), filter_rb, FALSE, FALSE, 0);
|
|
gtk_tooltips_set_tip (tooltips, filter_rb, ("Search for data by display filter syntax.\ne.g. ip.addr==10.1.1.1"), NULL);
|
|
gtk_widget_show(filter_rb);
|
|
|
|
/* Hex */
|
|
hex_rb = RADIO_BUTTON_NEW_WITH_MNEMONIC(filter_rb,
|
|
"_Hex value", accel_group);
|
|
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(hex_rb), cfile.hex);
|
|
gtk_box_pack_start(GTK_BOX(find_type_hb), hex_rb, FALSE, FALSE, 0);
|
|
gtk_tooltips_set_tip (tooltips, hex_rb, ("Search for data by hex string.\ne.g. fffffda5"), NULL);
|
|
gtk_widget_show(hex_rb);
|
|
|
|
/* ASCII Search */
|
|
string_rb = RADIO_BUTTON_NEW_WITH_MNEMONIC(filter_rb,
|
|
"_String", accel_group);
|
|
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(string_rb), cfile.string);
|
|
gtk_box_pack_start(GTK_BOX(find_type_hb), string_rb, FALSE, FALSE, 0);
|
|
gtk_tooltips_set_tip (tooltips, string_rb, ("Search for data by string value.\ne.g. My String"), NULL);
|
|
gtk_widget_show(string_rb);
|
|
|
|
/* Filter row */
|
|
filter_hb = gtk_hbox_new(FALSE, 3);
|
|
gtk_box_pack_start(GTK_BOX(find_type_vb), filter_hb, FALSE, FALSE, 0);
|
|
gtk_widget_show(filter_hb);
|
|
|
|
filter_bt = BUTTON_NEW_FROM_STOCK(ETHEREAL_STOCK_DISPLAY_FILTER_ENTRY);
|
|
SIGNAL_CONNECT(filter_bt, "clicked", display_filter_construct_cb, &args);
|
|
SIGNAL_CONNECT(filter_bt, "destroy", filter_button_destroy_cb, NULL);
|
|
OBJECT_SET_DATA(filter_bt, E_FILT_TE_BUTTON_KEY, filter_bt);
|
|
gtk_box_pack_start(GTK_BOX(filter_hb), filter_bt, FALSE, TRUE, 0);
|
|
gtk_tooltips_set_tip (tooltips, filter_bt, ("Click on the filter button to select a display filter,\nor enter your search criteria into the text box"), NULL);
|
|
gtk_widget_show(filter_bt);
|
|
|
|
filter_text_box = gtk_entry_new();
|
|
if (cfile.sfilter) gtk_entry_set_text(GTK_ENTRY(filter_text_box), cfile.sfilter);
|
|
OBJECT_SET_DATA(filter_bt, E_FILT_TE_PTR_KEY, filter_text_box);
|
|
OBJECT_SET_DATA(find_frame_w, E_FILT_TE_PTR_KEY, filter_text_box);
|
|
gtk_box_pack_start(GTK_BOX(filter_hb), filter_text_box, TRUE, TRUE, 0);
|
|
SIGNAL_CONNECT(filter_text_box, "changed", find_filter_te_syntax_check_cb, find_frame_w);
|
|
gtk_widget_show(filter_text_box);
|
|
|
|
|
|
/* */
|
|
main_options_hb = gtk_hbox_new(FALSE, 3);
|
|
gtk_container_add(GTK_CONTAINER(main_vb), main_options_hb);
|
|
gtk_widget_show(main_options_hb);
|
|
|
|
|
|
/* search in frame */
|
|
data_frame = gtk_frame_new("Search In");
|
|
gtk_box_pack_start(GTK_BOX(main_options_hb), data_frame, TRUE, TRUE, 0);
|
|
gtk_widget_show(data_frame);
|
|
|
|
/* search in row */
|
|
data_vb = gtk_vbox_new(TRUE, 0);
|
|
gtk_container_border_width(GTK_CONTAINER(data_vb), 3);
|
|
gtk_container_add(GTK_CONTAINER(data_frame), data_vb);
|
|
gtk_widget_show(data_vb);
|
|
|
|
/* Packet list */
|
|
summary_data_rb = RADIO_BUTTON_NEW_WITH_MNEMONIC(NULL,
|
|
"Packet list", accel_group);
|
|
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(summary_data_rb), summary_data);
|
|
gtk_box_pack_start(GTK_BOX(data_vb), summary_data_rb, TRUE, TRUE, 0);
|
|
gtk_tooltips_set_tip (tooltips, summary_data_rb, ("Search for string in the Info column of the packet summary (top pane)"), NULL);
|
|
gtk_widget_show(summary_data_rb);
|
|
|
|
/* Packet details */
|
|
decode_data_rb = RADIO_BUTTON_NEW_WITH_MNEMONIC(summary_data_rb,
|
|
"Packet details", accel_group);
|
|
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(decode_data_rb), decode_data);
|
|
gtk_box_pack_start(GTK_BOX(data_vb), decode_data_rb, TRUE, TRUE, 0);
|
|
gtk_tooltips_set_tip (tooltips, decode_data_rb, ("Search for string in the decoded packet display (middle pane)"), NULL);
|
|
gtk_widget_show(decode_data_rb);
|
|
|
|
/* Packet bytes */
|
|
hex_data_rb = RADIO_BUTTON_NEW_WITH_MNEMONIC(summary_data_rb,
|
|
"Packet bytes", accel_group);
|
|
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(hex_data_rb), !decode_data && !summary_data);
|
|
gtk_box_pack_start(GTK_BOX(data_vb), hex_data_rb, TRUE, TRUE, 0);
|
|
gtk_tooltips_set_tip (tooltips, hex_data_rb, ("Search for string in the packet data"), NULL);
|
|
gtk_widget_show(hex_data_rb);
|
|
|
|
/* string options frame */
|
|
string_opt_frame = gtk_frame_new("String Options");
|
|
gtk_box_pack_start(GTK_BOX(main_options_hb), string_opt_frame, TRUE, TRUE, 0);
|
|
gtk_widget_show(string_opt_frame);
|
|
|
|
string_opt_vb = gtk_vbox_new(FALSE, 0);
|
|
gtk_container_add(GTK_CONTAINER(string_opt_frame), string_opt_vb);
|
|
gtk_container_border_width(GTK_CONTAINER(string_opt_vb), 3);
|
|
gtk_widget_show(string_opt_vb);
|
|
|
|
case_cb = CHECK_BUTTON_NEW_WITH_MNEMONIC(
|
|
"Case sensitive", accel_group);
|
|
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(case_cb), !case_type);
|
|
gtk_container_add(GTK_CONTAINER(string_opt_vb), case_cb);
|
|
gtk_tooltips_set_tip (tooltips, case_cb, ("Search by mixed upper/lower case?"), NULL);
|
|
gtk_widget_show(case_cb);
|
|
|
|
combo_lb = gtk_label_new("Character set:");
|
|
gtk_container_add(GTK_CONTAINER(string_opt_vb), combo_lb);
|
|
gtk_misc_set_alignment(GTK_MISC(combo_lb), 0.0, 0.5);
|
|
gtk_widget_show(combo_lb);
|
|
|
|
/* String Type Selection Dropdown Box
|
|
These only apply to the Hex Window search option */
|
|
/* Create Combo Box */
|
|
combo_cb = gtk_combo_new();
|
|
|
|
glist = g_list_append(glist, "ASCII Unicode & Non-Unicode");
|
|
glist = g_list_append(glist, "ASCII Non-Unicode");
|
|
glist = g_list_append(glist, "ASCII Unicode");
|
|
|
|
gtk_combo_set_popdown_strings(GTK_COMBO(combo_cb), glist);
|
|
/* You only get to choose from the options we offer */
|
|
gtk_entry_set_editable(GTK_ENTRY(GTK_COMBO(combo_cb)->entry), FALSE);
|
|
gtk_container_add(GTK_CONTAINER(string_opt_vb), combo_cb);
|
|
gtk_widget_show(combo_cb);
|
|
|
|
|
|
/* direction frame */
|
|
direction_frame = gtk_frame_new("Direction");
|
|
gtk_box_pack_start(GTK_BOX(main_options_hb), direction_frame, FALSE, FALSE, 0);
|
|
gtk_widget_show(direction_frame);
|
|
|
|
/* Direction row: Forward and reverse radio buttons */
|
|
direction_vb = gtk_vbox_new(FALSE, 0);
|
|
gtk_container_border_width(GTK_CONTAINER(direction_vb), 3);
|
|
gtk_container_add(GTK_CONTAINER(direction_frame), direction_vb);
|
|
gtk_widget_show(direction_vb);
|
|
|
|
up_rb = RADIO_BUTTON_NEW_WITH_MNEMONIC(NULL, "_Up", accel_group);
|
|
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(up_rb), cfile.sbackward);
|
|
gtk_box_pack_start(GTK_BOX(direction_vb), up_rb, FALSE, FALSE, 0);
|
|
gtk_widget_show(up_rb);
|
|
|
|
down_rb = RADIO_BUTTON_NEW_WITH_MNEMONIC(up_rb, "_Down", accel_group);
|
|
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(down_rb), !cfile.sbackward);
|
|
gtk_box_pack_start(GTK_BOX(direction_vb), down_rb, FALSE, FALSE, 0);
|
|
gtk_widget_show(down_rb);
|
|
|
|
|
|
/* Button row */
|
|
if(topic_available(HELP_FIND_DIALOG)) {
|
|
bbox = dlg_button_row_new(GTK_STOCK_FIND, GTK_STOCK_CANCEL, GTK_STOCK_HELP, NULL);
|
|
} else {
|
|
bbox = dlg_button_row_new(GTK_STOCK_FIND, GTK_STOCK_CANCEL, NULL);
|
|
}
|
|
gtk_box_pack_start(GTK_BOX(main_vb), bbox, FALSE, FALSE, 0);
|
|
gtk_widget_show(bbox);
|
|
|
|
ok_bt = OBJECT_GET_DATA(bbox, GTK_STOCK_FIND);
|
|
SIGNAL_CONNECT(ok_bt, "clicked", find_frame_ok_cb, find_frame_w);
|
|
|
|
cancel_bt = OBJECT_GET_DATA(bbox, GTK_STOCK_CANCEL);
|
|
SIGNAL_CONNECT(cancel_bt, "clicked", find_frame_close_cb, find_frame_w);
|
|
|
|
if(topic_available(HELP_FIND_DIALOG)) {
|
|
help_bt = OBJECT_GET_DATA(bbox, GTK_STOCK_HELP);
|
|
SIGNAL_CONNECT(help_bt, "clicked", topic_cb, HELP_FIND_DIALOG);
|
|
}
|
|
|
|
/* Attach pointers to needed widgets to the capture prefs window/object */
|
|
OBJECT_SET_DATA(find_frame_w, E_FIND_FILT_KEY, filter_text_box);
|
|
OBJECT_SET_DATA(find_frame_w, E_FIND_BACKWARD_KEY, up_rb);
|
|
OBJECT_SET_DATA(find_frame_w, E_FIND_FILTERDATA_KEY, filter_rb);
|
|
OBJECT_SET_DATA(find_frame_w, E_FIND_HEXDATA_KEY, hex_rb);
|
|
OBJECT_SET_DATA(find_frame_w, E_FIND_STRINGDATA_KEY, string_rb);
|
|
OBJECT_SET_DATA(find_frame_w, E_FIND_STRINGTYPE_LABEL_KEY, combo_lb);
|
|
OBJECT_SET_DATA(find_frame_w, E_FIND_STRINGTYPE_KEY, combo_cb);
|
|
OBJECT_SET_DATA(find_frame_w, E_CASE_SEARCH_KEY, case_cb);
|
|
OBJECT_SET_DATA(find_frame_w, E_SOURCE_HEX_KEY, hex_data_rb);
|
|
OBJECT_SET_DATA(find_frame_w, E_SOURCE_DECODE_KEY, decode_data_rb);
|
|
OBJECT_SET_DATA(find_frame_w, E_SOURCE_SUMMARY_KEY, summary_data_rb);
|
|
OBJECT_SET_DATA(find_frame_w, E_FILT_TE_BUTTON_KEY, filter_bt);
|
|
|
|
/*
|
|
* Now that we've attached the pointers, connect the signals - if
|
|
* we do so before we've attached the pointers, the signals may
|
|
* be delivered before the pointers are attached; the signal
|
|
* handlers expect the pointers to be attached, and won't be happy.
|
|
*/
|
|
SIGNAL_CONNECT(hex_rb, "clicked", hex_selected_cb, find_frame_w);
|
|
SIGNAL_CONNECT(string_rb, "clicked", string_selected_cb, find_frame_w);
|
|
SIGNAL_CONNECT(filter_rb, "clicked", filter_selected_cb, find_frame_w);
|
|
|
|
string_selected_cb(NULL, find_frame_w);
|
|
filter_selected_cb(NULL, find_frame_w);
|
|
|
|
window_set_cancel_button(find_frame_w, cancel_bt, window_cancel_button_cb);
|
|
|
|
gtk_widget_grab_default(ok_bt);
|
|
|
|
/* Catch the "activate" signal on the filter text entry, so that
|
|
if the user types Return there, we act as if the "OK" button
|
|
had been selected, as happens if Return is typed if some widget
|
|
that *doesn't* handle the Return key has the input focus. */
|
|
dlg_set_activate(filter_text_box, ok_bt);
|
|
|
|
/* Give the initial focus to the "Filter" entry box. */
|
|
gtk_widget_grab_focus(filter_text_box);
|
|
|
|
SIGNAL_CONNECT(find_frame_w, "delete_event", window_delete_event_cb, NULL);
|
|
SIGNAL_CONNECT(find_frame_w, "destroy", find_frame_destroy_cb, NULL);
|
|
|
|
gtk_widget_show(find_frame_w);
|
|
window_present(find_frame_w);
|
|
}
|
|
|
|
/* this function opens the find frame dialogue and sets the filter string */
|
|
void
|
|
find_frame_with_filter(char *filter)
|
|
{
|
|
find_frame_cb(NULL, NULL);
|
|
gtk_entry_set_text(GTK_ENTRY(filter_text_box), filter);
|
|
}
|
|
|
|
/*
|
|
* Check the filter syntax based on the type of search we're doing.
|
|
*/
|
|
static void
|
|
find_filter_te_syntax_check_cb(GtkWidget *w, gpointer parent_w)
|
|
{
|
|
const gchar *strval;
|
|
GtkWidget *hex_rb, *string_rb;
|
|
guint8 *bytes = NULL;
|
|
size_t nbytes;
|
|
|
|
hex_rb = (GtkWidget *)OBJECT_GET_DATA(parent_w, E_FIND_HEXDATA_KEY);
|
|
string_rb = (GtkWidget *)OBJECT_GET_DATA(parent_w, E_FIND_STRINGDATA_KEY);
|
|
|
|
if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (hex_rb))) {
|
|
/*
|
|
* Hex search - scan the search string to make sure it's valid hex.
|
|
*/
|
|
strval = gtk_entry_get_text(GTK_ENTRY(w));
|
|
if (strval == NULL) {
|
|
/* XXX - can this happen? */
|
|
colorize_filter_te_as_invalid(w);
|
|
} else {
|
|
bytes = convert_string_to_hex(strval, &nbytes);
|
|
if (bytes == NULL)
|
|
colorize_filter_te_as_invalid(w);
|
|
else {
|
|
g_free(bytes);
|
|
colorize_filter_te_as_valid(w);
|
|
}
|
|
}
|
|
} else if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (string_rb))) {
|
|
/*
|
|
* String search. Make sure the string isn't empty.
|
|
*/
|
|
strval = gtk_entry_get_text(GTK_ENTRY(w));
|
|
if (strval == NULL) {
|
|
/* XXX - can this happen? */
|
|
colorize_filter_te_as_invalid(w);
|
|
} else {
|
|
if (strcmp(strval, "") == 0)
|
|
colorize_filter_te_as_invalid(w);
|
|
else
|
|
colorize_filter_te_as_valid(w);
|
|
}
|
|
} else {
|
|
/*
|
|
* Display filter search; check it with "filter_te_syntax_check_cb()".
|
|
*/
|
|
filter_te_syntax_check_cb(w);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* This function will re-check the search text syntax.
|
|
*/
|
|
static void
|
|
hex_selected_cb(GtkWidget *button_rb _U_, gpointer parent_w)
|
|
{
|
|
GtkWidget *filter_text_box;
|
|
|
|
filter_text_box = (GtkWidget *) OBJECT_GET_DATA(parent_w, E_FILT_TE_PTR_KEY);
|
|
|
|
/* Re-check the display filter. */
|
|
find_filter_te_syntax_check_cb(filter_text_box, parent_w);
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* This function will disable the string options until
|
|
* the string search is selected.
|
|
*/
|
|
static void
|
|
string_selected_cb(GtkWidget *button_rb _U_, gpointer parent_w)
|
|
{
|
|
GtkWidget *string_rb, *hex_data_rb, *decode_data_rb, *summary_data_rb,
|
|
*data_combo_lb, *data_combo_cb, *data_case_cb, *filter_text_box;
|
|
|
|
string_rb = (GtkWidget *)OBJECT_GET_DATA(parent_w, E_FIND_STRINGDATA_KEY);
|
|
hex_data_rb = (GtkWidget *)OBJECT_GET_DATA(parent_w, E_SOURCE_HEX_KEY);
|
|
decode_data_rb = (GtkWidget *)OBJECT_GET_DATA(parent_w, E_SOURCE_DECODE_KEY);
|
|
summary_data_rb = (GtkWidget *)OBJECT_GET_DATA(parent_w, E_SOURCE_SUMMARY_KEY);
|
|
|
|
data_combo_lb = (GtkWidget *)OBJECT_GET_DATA(parent_w, E_FIND_STRINGTYPE_LABEL_KEY);
|
|
data_combo_cb = (GtkWidget *)OBJECT_GET_DATA(parent_w, E_FIND_STRINGTYPE_KEY);
|
|
data_case_cb = (GtkWidget *) OBJECT_GET_DATA(parent_w, E_CASE_SEARCH_KEY);
|
|
filter_text_box = (GtkWidget *) OBJECT_GET_DATA(parent_w, E_FILT_TE_PTR_KEY);
|
|
|
|
if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(string_rb))) {
|
|
gtk_widget_set_sensitive(GTK_WIDGET(hex_data_rb), TRUE);
|
|
gtk_widget_set_sensitive(GTK_WIDGET(decode_data_rb), TRUE);
|
|
gtk_widget_set_sensitive(GTK_WIDGET(summary_data_rb), TRUE);
|
|
gtk_widget_set_sensitive(GTK_WIDGET(data_combo_lb), TRUE);
|
|
gtk_widget_set_sensitive(GTK_WIDGET(data_combo_cb), TRUE);
|
|
gtk_widget_set_sensitive(GTK_WIDGET(data_case_cb), TRUE);
|
|
} else {
|
|
gtk_widget_set_sensitive(GTK_WIDGET(hex_data_rb), FALSE);
|
|
gtk_widget_set_sensitive(GTK_WIDGET(decode_data_rb), FALSE);
|
|
gtk_widget_set_sensitive(GTK_WIDGET(summary_data_rb), FALSE);
|
|
gtk_widget_set_sensitive(GTK_WIDGET(data_combo_lb), FALSE);
|
|
gtk_widget_set_sensitive(GTK_WIDGET(data_combo_cb), FALSE);
|
|
gtk_widget_set_sensitive(GTK_WIDGET(data_case_cb), FALSE);
|
|
}
|
|
/* Re-check the display filter. */
|
|
find_filter_te_syntax_check_cb(filter_text_box, parent_w);
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* This function will disable the filter button until
|
|
* the filter search is selected.
|
|
*/
|
|
static void
|
|
filter_selected_cb(GtkWidget *button_rb _U_, gpointer parent_w)
|
|
{
|
|
GtkWidget *filter_bt, *filter_rb;
|
|
|
|
filter_bt = (GtkWidget *)OBJECT_GET_DATA(parent_w, E_FILT_TE_BUTTON_KEY);
|
|
filter_rb = (GtkWidget *)OBJECT_GET_DATA(parent_w, E_FIND_FILTERDATA_KEY);
|
|
|
|
if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(filter_rb)))
|
|
{
|
|
gtk_widget_set_sensitive(GTK_WIDGET(filter_bt), TRUE);
|
|
}
|
|
else
|
|
{
|
|
gtk_widget_set_sensitive(GTK_WIDGET(filter_bt), FALSE);
|
|
}
|
|
return;
|
|
}
|
|
|
|
static void
|
|
find_frame_ok_cb(GtkWidget *ok_bt _U_, gpointer parent_w)
|
|
{
|
|
GtkWidget *filter_te, *up_rb, *hex_rb, *string_rb, *combo_cb,
|
|
*case_cb, *decode_data_rb, *summary_data_rb;
|
|
const gchar *filter_text, *string_type;
|
|
search_charset_t scs_type = SCS_ASCII_AND_UNICODE;
|
|
guint8 *bytes = NULL;
|
|
size_t nbytes;
|
|
char *string = NULL;
|
|
dfilter_t *sfcode;
|
|
gboolean found_packet;
|
|
|
|
filter_te = (GtkWidget *)OBJECT_GET_DATA(parent_w, E_FIND_FILT_KEY);
|
|
up_rb = (GtkWidget *)OBJECT_GET_DATA(parent_w, E_FIND_BACKWARD_KEY);
|
|
hex_rb = (GtkWidget *)OBJECT_GET_DATA(parent_w, E_FIND_HEXDATA_KEY);
|
|
string_rb = (GtkWidget *)OBJECT_GET_DATA(parent_w, E_FIND_STRINGDATA_KEY);
|
|
combo_cb = (GtkWidget *)OBJECT_GET_DATA(parent_w, E_FIND_STRINGTYPE_KEY);
|
|
case_cb = (GtkWidget *) OBJECT_GET_DATA(parent_w, E_CASE_SEARCH_KEY);
|
|
decode_data_rb = (GtkWidget *)OBJECT_GET_DATA(parent_w, E_SOURCE_DECODE_KEY);
|
|
summary_data_rb = (GtkWidget *)OBJECT_GET_DATA(parent_w, E_SOURCE_SUMMARY_KEY);
|
|
|
|
filter_text = gtk_entry_get_text(GTK_ENTRY(filter_te));
|
|
string_type = gtk_entry_get_text(GTK_ENTRY(GTK_COMBO(combo_cb)->entry));
|
|
|
|
case_type = !gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(case_cb));
|
|
decode_data = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(decode_data_rb));
|
|
summary_data = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(summary_data_rb));
|
|
|
|
/*
|
|
* Process the search criterion.
|
|
*/
|
|
if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (hex_rb))) {
|
|
/*
|
|
* Hex search - scan the search string to make sure it's valid hex
|
|
* and to find out how many bytes there are.
|
|
*/
|
|
bytes = convert_string_to_hex(filter_text, &nbytes);
|
|
if (bytes == NULL) {
|
|
simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK,
|
|
"You didn't specify a valid hex string.");
|
|
return;
|
|
}
|
|
} else if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (string_rb))) {
|
|
/*
|
|
* String search.
|
|
* Make sure we're searching for something, first.
|
|
*/
|
|
if (strcmp(filter_text, "") == 0) {
|
|
simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK,
|
|
"You didn't specify any text for which to search.");
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* We are - get the character set type.
|
|
*/
|
|
if (strcmp(string_type, "ASCII Unicode & Non-Unicode") == 0)
|
|
scs_type = SCS_ASCII_AND_UNICODE;
|
|
else if (strcmp(string_type, "ASCII Non-Unicode") == 0)
|
|
scs_type = SCS_ASCII;
|
|
else if (strcmp(string_type, "ASCII Unicode") == 0)
|
|
scs_type = SCS_UNICODE;
|
|
else {
|
|
simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, "You didn't choose a valid character set.");
|
|
return;
|
|
}
|
|
string = convert_string_case(filter_text, case_type);
|
|
} else {
|
|
/*
|
|
* Display filter search - try to compile the filter.
|
|
*/
|
|
if (!dfilter_compile(filter_text, &sfcode)) {
|
|
/* The attempt failed; report an error. */
|
|
bad_dfilter_alert_box(filter_text);
|
|
return;
|
|
}
|
|
|
|
/* Was it empty? */
|
|
if (sfcode == NULL) {
|
|
/* Yes - complain. */
|
|
simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK,
|
|
"You specified a filter that doesn't test anything.");
|
|
return;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Remember the search parameters.
|
|
*/
|
|
if (cfile.sfilter)
|
|
g_free(cfile.sfilter);
|
|
cfile.sfilter = g_strdup(filter_text);
|
|
cfile.sbackward = GTK_TOGGLE_BUTTON (up_rb)->active;
|
|
cfile.hex = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (hex_rb));
|
|
cfile.string = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (string_rb));
|
|
cfile.scs_type = scs_type;
|
|
cfile.case_type = case_type;
|
|
cfile.decode_data = decode_data;
|
|
cfile.summary_data = summary_data;
|
|
|
|
if (cfile.hex) {
|
|
found_packet = cf_find_packet_data(&cfile, bytes, nbytes);
|
|
g_free(bytes);
|
|
if (!found_packet) {
|
|
/* We didn't find a packet */
|
|
simple_dialog(ESD_TYPE_INFO, ESD_BTN_OK,
|
|
"%sFound no match!%s\n\n"
|
|
"No packet contained those bytes.",
|
|
simple_dialog_primary_start(), simple_dialog_primary_end());
|
|
return;
|
|
}
|
|
} else if (cfile.string) {
|
|
/* OK, what are we searching? */
|
|
if (cfile.decode_data) {
|
|
/* The text in the protocol tree */
|
|
found_packet = cf_find_packet_protocol_tree(&cfile, string);
|
|
g_free(string);
|
|
if (!found_packet) {
|
|
/* We didn't find the packet. */
|
|
simple_dialog(ESD_TYPE_INFO, ESD_BTN_OK,
|
|
"%sFound no match!%s\n\n"
|
|
"No packet contained that string in its dissected display.",
|
|
simple_dialog_primary_start(), simple_dialog_primary_end());
|
|
return;
|
|
}
|
|
} else if (cfile.summary_data) {
|
|
/* The text in the summary line */
|
|
found_packet = cf_find_packet_summary_line(&cfile, string);
|
|
g_free(string);
|
|
if (!found_packet) {
|
|
/* We didn't find the packet. */
|
|
simple_dialog(ESD_TYPE_INFO, ESD_BTN_OK,
|
|
"%sFound no match!%s\n\n"
|
|
"No packet contained that string in its Info column.",
|
|
simple_dialog_primary_start(), simple_dialog_primary_end());
|
|
return;
|
|
}
|
|
} else {
|
|
/* The raw packet data */
|
|
found_packet = cf_find_packet_data(&cfile, string, strlen(string));
|
|
g_free(string);
|
|
if (!found_packet) {
|
|
/* We didn't find the packet. */
|
|
simple_dialog(ESD_TYPE_INFO, ESD_BTN_OK,
|
|
"%sFound no match!%s\n\n"
|
|
"No packet contained that string in its data.",
|
|
simple_dialog_primary_start(), simple_dialog_primary_end());
|
|
return;
|
|
}
|
|
}
|
|
} else {
|
|
found_packet = cf_find_packet_dfilter(&cfile, sfcode);
|
|
dfilter_free(sfcode);
|
|
if (!found_packet) {
|
|
/* We didn't find a packet */
|
|
simple_dialog(ESD_TYPE_INFO, ESD_BTN_OK,
|
|
"%sFound no match!%s\n\n"
|
|
"No packet matched that filter.",
|
|
simple_dialog_primary_start(), simple_dialog_primary_end());
|
|
g_free(bytes);
|
|
return;
|
|
}
|
|
}
|
|
window_destroy(GTK_WIDGET(parent_w));
|
|
}
|
|
|
|
static void
|
|
find_frame_close_cb(GtkWidget *close_bt _U_, gpointer parent_w)
|
|
{
|
|
gtk_grab_remove(GTK_WIDGET(parent_w));
|
|
window_destroy(GTK_WIDGET(parent_w));
|
|
}
|
|
|
|
static void
|
|
find_frame_destroy_cb(GtkWidget *win _U_, gpointer user_data _U_)
|
|
{
|
|
/* Note that we no longer have a "Find Packet" dialog box. */
|
|
find_frame_w = NULL;
|
|
}
|
|
|
|
static void
|
|
find_previous_next(GtkWidget *w, gpointer d, gboolean sens)
|
|
{
|
|
guint8 *bytes;
|
|
size_t nbytes;
|
|
char *string;
|
|
dfilter_t *sfcode;
|
|
|
|
if (cfile.sfilter) {
|
|
cfile.sbackward = sens;
|
|
if (cfile.hex) {
|
|
bytes = convert_string_to_hex(cfile.sfilter, &nbytes);
|
|
if (bytes == NULL) {
|
|
/*
|
|
* XXX - this shouldn't happen, as we've already successfully
|
|
* translated the string once.
|
|
*/
|
|
return;
|
|
}
|
|
cf_find_packet_data(&cfile, bytes, nbytes);
|
|
g_free(bytes);
|
|
} else if (cfile.string) {
|
|
string = convert_string_case(cfile.sfilter, cfile.case_type);
|
|
/* OK, what are we searching? */
|
|
if (cfile.decode_data) {
|
|
/* The text in the protocol tree */
|
|
cf_find_packet_protocol_tree(&cfile, string);
|
|
} else if (cfile.summary_data) {
|
|
/* The text in the summary line */
|
|
cf_find_packet_summary_line(&cfile, string);
|
|
} else {
|
|
/* The raw packet data */
|
|
cf_find_packet_data(&cfile, string, strlen(string));
|
|
}
|
|
g_free(string);
|
|
} else {
|
|
if (!dfilter_compile(cfile.sfilter, &sfcode)) {
|
|
/*
|
|
* XXX - this shouldn't happen, as we've already successfully
|
|
* translated the string once.
|
|
*/
|
|
return;
|
|
}
|
|
if (sfcode == NULL) {
|
|
/*
|
|
* XXX - this shouldn't happen, as we've already found that the
|
|
* string wasn't null.
|
|
*/
|
|
return;
|
|
}
|
|
cf_find_packet_dfilter(&cfile, sfcode);
|
|
dfilter_free(sfcode);
|
|
}
|
|
} else
|
|
find_frame_cb(w, d);
|
|
}
|
|
|
|
void
|
|
find_next_cb(GtkWidget *w , gpointer d)
|
|
{
|
|
find_previous_next(w, d, FALSE);
|
|
}
|
|
|
|
void
|
|
find_previous_cb(GtkWidget *w , gpointer d)
|
|
{
|
|
find_previous_next(w, d, TRUE);
|
|
}
|
|
|
|
/* this function jumps to the next packet matching the filter */
|
|
void
|
|
find_previous_next_frame_with_filter(const char *filter, gboolean backwards)
|
|
{
|
|
dfilter_t *sfcode;
|
|
gboolean sbackwards_saved;
|
|
|
|
/* temporarily set the direction we want to search */
|
|
sbackwards_saved=cfile.sbackward;
|
|
cfile.sbackward = backwards;
|
|
|
|
if (!dfilter_compile(filter, &sfcode)) {
|
|
/*
|
|
* XXX - this shouldn't happen, as the filter string is machine
|
|
* generated
|
|
*/
|
|
return;
|
|
}
|
|
if (sfcode == NULL) {
|
|
/*
|
|
* XXX - this shouldn't happen, as the filter string is machine
|
|
* generated.
|
|
*/
|
|
return;
|
|
}
|
|
cf_find_packet_dfilter(&cfile, sfcode);
|
|
dfilter_free(sfcode);
|
|
cfile.sbackward=sbackwards_saved;
|
|
}
|