aboutsummaryrefslogtreecommitdiffstats
path: root/ui/gtk/capture_info_dlg.c
diff options
context:
space:
mode:
authorJörg Mayer <jmayer@loplof.de>2012-01-15 21:59:11 +0000
committerJörg Mayer <jmayer@loplof.de>2012-01-15 21:59:11 +0000
commitbe706c63801fb98d42fb743b27b16cc36273651e (patch)
tree62ed0b552191eb0753d26a3edcbab73459a15f7f /ui/gtk/capture_info_dlg.c
parent6d69ef093cd6868ab51f8b52477a510172033353 (diff)
Move gtk to ui/gtk.
This looses the last checkin to gtk, will add this manually back. svn path=/trunk/; revision=40518
Diffstat (limited to 'ui/gtk/capture_info_dlg.c')
-rw-r--r--ui/gtk/capture_info_dlg.c371
1 files changed, 371 insertions, 0 deletions
diff --git a/ui/gtk/capture_info_dlg.c b/ui/gtk/capture_info_dlg.c
new file mode 100644
index 0000000000..d99724fc51
--- /dev/null
+++ b/ui/gtk/capture_info_dlg.c
@@ -0,0 +1,371 @@
+/* capture_info_dlg.c
+ * Routines for packet capture info dialog
+ *
+ * $Id$
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * 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
+
+#ifdef HAVE_LIBPCAP
+
+#ifdef HAVE_SYS_TYPES_H
+# include <sys/types.h>
+#endif
+
+#include <string.h>
+#include <time.h>
+
+#include <gtk/gtk.h>
+
+#include <epan/packet.h>
+
+#include "../capture.h"
+#include "../capture_info.h"
+#include "../capture_ui_utils.h"
+#include "../capture-pcap-util.h"
+
+#include "ui/gtk/dlg_utils.h"
+#include "ui/gtk/gui_utils.h"
+#include "ui/gtk/main.h"
+#include "ui/gtk/help_dlg.h"
+#include "ui/gtk/stock_icons.h"
+
+#ifdef HAVE_AIRPCAP
+#include <airpcap.h>
+#include "airpcap_loader.h"
+#include "airpcap_gui_utils.h"
+#include "airpcap_dlg.h"
+#endif
+
+
+/* a single capture counter value (with title, pointer to value and GtkWidgets) */
+/* as the packet_counts is a struct, not an array, keep a pointer to the */
+/* corresponding value packet_counts, to speed up (and simplify) output of values */
+typedef struct {
+ const gchar *title;
+ gint *value_ptr;
+ GtkWidget *label, *value_lb, *percent_pb, *percent_lb;
+} capture_info_counts_t;
+
+/* all data we need to know of this dialog, after creation finished */
+typedef struct {
+ GtkWidget *cap_w;
+ GtkWidget *running_time_lb;
+ capture_info_counts_t counts[PACKET_COUNTS_SIZE];
+ guint timer_id;
+ time_t start_time;
+} capture_info_ui_t;
+
+
+/* Workhorse for stopping capture */
+static void
+capture_info_stop(capture_options *capture_opts)
+{
+#ifdef HAVE_AIRPCAP
+ airpcap_set_toolbar_stop_capture(airpcap_if_active);
+#endif
+ capture_stop(capture_opts);
+}
+
+/* "delete-event" signal callback. Note different signature than "clicked" signal callback */
+static gboolean
+capture_info_delete_cb(GtkWidget *w _U_, GdkEvent *event _U_, gpointer data) {
+ capture_info_stop(data);
+ return TRUE;
+}
+
+/* "clicked" signal callback */
+static void
+capture_info_stop_clicked_cb(GtkButton *w _U_, gpointer data) {
+ capture_info_stop(data);
+}
+
+static gboolean
+capture_info_ui_update_cb(gpointer data)
+{
+ capture_info *cinfo = data;
+ capture_info_ui_t *info = cinfo->ui;
+
+ if (!info) /* ...which might happen on slow displays? */
+ return TRUE;
+
+ cinfo->running_time = time(NULL) - info->start_time;
+ capture_info_ui_update(cinfo);
+ return TRUE; /* call the timer again */
+}
+
+
+/* create the capture info dialog */
+/* will keep pointers to the fields in the counts parameter */
+void capture_info_ui_create(
+capture_info *cinfo,
+capture_options *capture_opts)
+{
+ unsigned int i;
+ GtkWidget *main_vb, *stop_bt, *counts_tb;
+ GtkWidget *counts_fr, *running_tb, *running_label, *bbox, *ci_help;
+ capture_info_ui_t *info;
+ gchar *cap_w_title;
+ gchar *title_iface;
+ gchar *descr;
+ GString *str;
+
+ info = g_malloc0(sizeof(capture_info_ui_t));
+ info->counts[0].title = "Total";
+ info->counts[0].value_ptr = &(cinfo->counts->total);
+ info->counts[1].title = "SCTP";
+ info->counts[1].value_ptr = &(cinfo->counts->sctp);
+ info->counts[2].title = "TCP";
+ info->counts[2].value_ptr = &(cinfo->counts->tcp);
+ info->counts[3].title = "UDP";
+ info->counts[3].value_ptr = &(cinfo->counts->udp);
+ info->counts[4].title = "ICMP";
+ info->counts[4].value_ptr = &(cinfo->counts->icmp);
+ info->counts[5].title = "ARP";
+ info->counts[5].value_ptr = &(cinfo->counts->arp);
+ info->counts[6].title = "OSPF";
+ info->counts[6].value_ptr = &(cinfo->counts->ospf);
+ info->counts[7].title = "GRE";
+ info->counts[7].value_ptr = &(cinfo->counts->gre);
+ info->counts[8].title = "NetBIOS";
+ info->counts[8].value_ptr = &(cinfo->counts->netbios);
+ info->counts[9].title = "IPX";
+ info->counts[9].value_ptr = &(cinfo->counts->ipx);
+ info->counts[10].title = "VINES";
+ info->counts[10].value_ptr = &(cinfo->counts->vines);
+ info->counts[11].title = "Other";
+ info->counts[11].value_ptr = &(cinfo->counts->other);
+ info->counts[12].title = "I2C Events";
+ info->counts[12].value_ptr = &(cinfo->counts->i2c_event);
+ info->counts[13].title = "I2C Data";
+ info->counts[13].value_ptr = &(cinfo->counts->i2c_data);
+
+ /*
+ * Create the dialog window, with a title that includes the interface.
+ *
+ * If we have a descriptive name for the interface, show that,
+ * rather than its raw name. On NT 5.x (2K/XP/Server2K3), the
+ * interface name is something like "\Device\NPF_{242423..."
+ * which is pretty useless to the normal user. On other platforms,
+ * it might be less cryptic, but if a more descriptive name is
+ * available, we should still use that.
+ */
+ str = g_string_new("");
+#ifdef _WIN32
+ if (capture_opts->ifaces->len < 2) {
+#else
+ if (capture_opts->ifaces->len < 4) {
+#endif
+ for (i = 0; i < capture_opts->ifaces->len; i++) {
+ interface_options interface_opts;
+
+ interface_opts = g_array_index(capture_opts->ifaces, interface_options, i);
+ descr = get_interface_descriptive_name(interface_opts.name);
+ if (i > 0) {
+ if (capture_opts->ifaces->len > 2) {
+ g_string_append_printf(str, ",");
+ }
+ g_string_append_printf(str, " ");
+ if (i == capture_opts->ifaces->len - 1) {
+ g_string_append_printf(str, "and ");
+ }
+ }
+ g_string_append_printf(str, "%s", descr);
+ g_free(descr);
+ }
+ } else {
+ g_string_append_printf(str, "%u interfaces", capture_opts->ifaces->len);
+ }
+ title_iface = g_strdup_printf("Wireshark: Capture from %s", str->str);
+ g_string_free(str, TRUE);
+ cap_w_title = create_user_window_title(title_iface);
+ g_free(title_iface);
+ info->cap_w = dlg_window_new(cap_w_title);
+ g_free(cap_w_title);
+
+ /* Container for capture display widgets */
+ main_vb = gtk_vbox_new(FALSE, 1);
+ gtk_container_set_border_width(GTK_CONTAINER(main_vb), 5);
+ gtk_container_add(GTK_CONTAINER(info->cap_w), main_vb);
+ gtk_widget_show(main_vb);
+
+ counts_fr = gtk_frame_new("Captured Packets");
+ gtk_box_pack_start(GTK_BOX(main_vb), counts_fr, FALSE, FALSE, 3);
+ gtk_widget_show(counts_fr);
+
+ /* Individual statistic elements */
+ counts_tb = gtk_table_new(PACKET_COUNTS_SIZE, 4, TRUE);
+ gtk_container_add(GTK_CONTAINER(counts_fr), counts_tb);
+ gtk_container_set_border_width(GTK_CONTAINER(counts_tb), 5);
+ gtk_widget_show(counts_tb);
+
+ gtk_table_set_row_spacings(GTK_TABLE(counts_tb), 0);
+ gtk_table_set_col_spacings(GTK_TABLE(counts_tb), 5);
+
+ for (i = 0; i < PACKET_COUNTS_SIZE; i++) {
+ info->counts[i].label = gtk_label_new(info->counts[i].title);
+ gtk_misc_set_alignment(GTK_MISC(info->counts[i].label), 0.0f, 0.5f);
+
+ info->counts[i].value_lb = gtk_label_new("0");
+ gtk_misc_set_alignment(GTK_MISC(info->counts[i].value_lb), 0.5f, 0.5f);
+
+ if (i == 0) {
+ /* do not build a progress bar for the "total" row */
+ /* (as this could suggest a "buffer full" to the user) */
+ /* simply put a label here */
+ info->counts[i].percent_pb = gtk_label_new("% of total");
+ } else {
+ /* build a progress bar in the other rows */
+ info->counts[i].percent_pb = gtk_progress_bar_new();
+
+ /* downsize the default size of this progress bar in x direction (def:150), */
+ /* otherwise it will become too large and the dialog will look ugly */
+ /* XXX: use a TreeView instead of a table in order to fix this */
+ gtk_widget_set_size_request(info->counts[i].percent_pb, 70, -1);
+ }
+
+ info->counts[i].percent_lb = gtk_label_new("0.0%");
+ gtk_misc_set_alignment(GTK_MISC(info->counts[i].percent_lb), 1.0f, 0.5f);
+
+ gtk_table_attach_defaults(GTK_TABLE(counts_tb),
+ info->counts[i].label, 0, 1, i, i + 1);
+ gtk_table_attach_defaults(GTK_TABLE(counts_tb),
+ info->counts[i].value_lb, 1, 2, i, i + 1);
+ gtk_table_attach_defaults(GTK_TABLE(counts_tb),
+ info->counts[i].percent_pb, 2, 3, i, i + 1);
+ gtk_table_attach_defaults(GTK_TABLE(counts_tb),
+ info->counts[i].percent_lb, 3, 4, i, i + 1);
+
+ gtk_widget_show(info->counts[i].label);
+ gtk_widget_show(info->counts[i].value_lb);
+ gtk_widget_show(info->counts[i].percent_pb);
+ /* don't show percentages for the "total" row */
+ if (i != 0) {
+ gtk_widget_show(info->counts[i].percent_lb);
+ }
+ }
+
+ /* Running time */
+ running_tb = gtk_table_new(1, 4, TRUE);
+ gtk_box_pack_start(GTK_BOX(main_vb), running_tb, FALSE, FALSE, 3);
+ gtk_widget_show(running_tb);
+
+ running_label = gtk_label_new("Running");
+ gtk_misc_set_alignment(GTK_MISC(running_label), 0.0f, 0.0f);
+ gtk_widget_show(running_label);
+ gtk_table_attach_defaults(GTK_TABLE(running_tb),
+ running_label, 0, 1, 0, 1);
+
+ info->running_time_lb = gtk_label_new("00:00:00");
+ gtk_misc_set_alignment(GTK_MISC(info->running_time_lb), 0.0f, 0.0f);
+ gtk_widget_show(info->running_time_lb);
+ gtk_table_attach(GTK_TABLE(running_tb),
+ info->running_time_lb,
+ 1, 2, 0, 1, 0, 0, 5, 0);
+
+ /* allow user to either click a stop button, or the close button on
+ the window to stop a capture in progress. */
+ bbox = dlg_button_row_new(WIRESHARK_STOCK_CAPTURE_STOP, GTK_STOCK_HELP, NULL);
+ gtk_box_pack_start(GTK_BOX(main_vb), bbox, FALSE, FALSE, 3);
+ gtk_widget_show(bbox);
+
+ stop_bt = g_object_get_data(G_OBJECT(bbox), WIRESHARK_STOCK_CAPTURE_STOP);
+ window_set_cancel_button(info->cap_w, stop_bt, NULL);
+ g_signal_connect(stop_bt, "clicked", G_CALLBACK(capture_info_stop_clicked_cb), capture_opts);
+ g_signal_connect(info->cap_w, "delete_event", G_CALLBACK(capture_info_delete_cb), capture_opts);
+
+ ci_help = g_object_get_data(G_OBJECT(bbox), GTK_STOCK_HELP);
+ gtk_widget_set_tooltip_text(ci_help, "Get help about this dialog");
+ g_signal_connect(ci_help, "clicked", G_CALLBACK(topic_cb), (gpointer)HELP_CAPTURE_INFO_DIALOG);
+
+ gtk_widget_show(info->cap_w);
+ window_present(info->cap_w);
+
+ info->start_time = time(NULL);
+
+ cinfo->ui = info;
+
+ /* update the dialog once a second, even if no packets rushing in */
+ info->timer_id = g_timeout_add(1000, capture_info_ui_update_cb,cinfo);
+}
+
+
+/* update the capture info dialog */
+/* As this function is a bit time critical while capturing, */
+/* prepare everything possible in the capture_info_ui_create() function above! */
+void capture_info_ui_update(
+capture_info *cinfo)
+{
+ unsigned int i;
+ gchar label_str[64];
+ capture_info_ui_t *info = cinfo->ui;
+
+ if (!info) /* ...which might happen on slow displays? */
+ return;
+
+ /* display running time */
+ g_snprintf(label_str, sizeof(label_str), "%02ld:%02ld:%02ld",
+ (long)(cinfo->running_time/3600), (long)((cinfo->running_time%3600)/60),
+ (long)(cinfo->running_time%60));
+ gtk_label_set_text(GTK_LABEL(info->running_time_lb), label_str);
+
+ /* if we have new packets, update all rows */
+ if (cinfo->new_packets) {
+ float pb_frac;
+ for (i = 0; i < PACKET_COUNTS_SIZE; i++) {
+ g_snprintf(label_str, sizeof(label_str), "%d", *info->counts[i].value_ptr);
+ gtk_label_set_text(GTK_LABEL(info->counts[i].value_lb), label_str);
+
+ pb_frac = (*info->counts[0].value_ptr != 0) ?
+ ((float)*info->counts[i].value_ptr / *info->counts[0].value_ptr) : 0.0f;
+
+ /* don't try to update the "total" row progress bar */
+ if (i != 0) {
+ gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(info->counts[i].percent_pb), pb_frac);
+ g_snprintf(label_str, sizeof(label_str), "%.1f%%", pb_frac * 100.0);
+ gtk_label_set_text(GTK_LABEL(info->counts[i].percent_lb), label_str);
+ }
+ }
+ }
+}
+
+/* destroy the capture info dialog again */
+void capture_info_ui_destroy(
+capture_info *cinfo)
+{
+ capture_info_ui_t *info = cinfo->ui;
+
+ if (!info) /* ...which probably shouldn't happen */
+ return;
+
+ g_source_remove(info->timer_id);
+
+ /* called from capture engine, so it's ok to destroy the dialog here */
+ gtk_grab_remove(GTK_WIDGET(info->cap_w));
+ window_destroy(GTK_WIDGET(info->cap_w));
+ g_free(info);
+ cinfo->ui = NULL;
+}
+
+
+#endif /* HAVE_LIBPCAP */