2 * GTK Tap implementation of stats_tree
3 * 2005, Luis E. G. Ontanon
7 * Wireshark - Network traffic analyzer
8 * By Gerald Combs <gerald@wireshark.org>
9 * Copyright 1998 Gerald Combs
11 * This program is free software; you can redistribute it and/or
12 * modify it under the terms of the GNU General Public License
13 * as published by the Free Software Foundation; either version 2
14 * of the License, or (at your option) any later version.
16 * This program is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 * GNU General Public License for more details.
21 * You should have received a copy of the GNU General Public License
22 * along with this program; if not, write to the Free Software
23 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
33 #include <epan/stats_tree_priv.h>
34 #include <epan/report_err.h>
36 #include "../simple_dialog.h"
37 #include "../globals.h"
38 #include "../stat_menu.h"
40 #include "gtk/gui_utils.h"
41 #include "gtk/dlg_utils.h"
42 #include "gtk/tap_dfilter_dlg.h"
46 struct _st_node_pres {
50 struct _tree_cfg_pres {
51 tap_dfilter_dlg* stat_dlg;
61 /* the columns of the tree pane */
62 enum _stat_tree_columns {
70 /* used for converting numbers */
71 #define NUM_BUF_SIZE 32
73 /* creates the gtk representation for a stat_node
77 setup_gtk_node_pr(stat_node* node)
79 GtkTreeIter* parent = NULL;
81 node->pr = g_malloc(sizeof(st_node_pres));
83 if (node->st->pr->store) {
84 node->pr->iter = g_malloc0(sizeof(GtkTreeIter));
86 if ( node->parent && node->parent->pr ) {
87 parent = node->parent->pr->iter;
89 gtk_tree_store_append (node->st->pr->store, node->pr->iter, parent);
90 gtk_tree_store_set(node->st->pr->store, node->pr->iter, TITLE_COLUMN, node->name, RATE_COLUMN, "", COUNT_COLUMN, "", -1);
96 draw_gtk_node(stat_node* node)
98 static gchar value[NUM_BUF_SIZE];
99 static gchar rate[NUM_BUF_SIZE];
100 static gchar percent[NUM_BUF_SIZE];
103 stats_tree_get_strs_from_node(node, value, rate,
106 if (node->st->pr->store && node->pr->iter) {
107 gtk_tree_store_set(node->st->pr->store, node->pr->iter,
110 PERCENT_COLUMN, percent,
114 if (node->children) {
115 for (child = node->children; child; child = child->next )
116 draw_gtk_node(child);
121 draw_gtk_tree(void *psp)
123 stats_tree *st = psp;
126 for (child = st->root.children; child; child = child->next ) {
127 draw_gtk_node(child);
129 if (child->pr->iter && st->pr->store) {
130 gtk_tree_view_expand_row(GTK_TREE_VIEW(st->pr->tree),
131 gtk_tree_model_get_path(GTK_TREE_MODEL(st->pr->store),
140 free_gtk_tree(GtkWindow *win _U_, stats_tree *st)
143 protect_thread_critical_region();
144 remove_tap_listener(st);
145 unprotect_thread_critical_region();
148 st->root.pr->iter = NULL;
150 st->cfg->in_use = FALSE;
156 clear_node_pr(stat_node* n)
159 for (c = n->children; c; c = c->next) {
164 gtk_tree_store_remove(n->st->pr->store, n->pr->iter);
174 for (c = st->root.children; c; c = c->next) {
181 /* initializes the stats_tree window */
183 init_gtk_tree(const char* optarg, void *userdata _U_)
185 gchar *abbr = stats_tree_get_abbr(optarg);
186 stats_tree* st = NULL;
187 stats_tree_cfg* cfg = NULL;
188 tree_pres* pr = g_malloc(sizeof(tree_pres));
190 gchar* window_name = NULL;
191 GString* error_string;
194 GtkWidget *main_vb, *bbox, *bt_close;
195 GtkTreeViewColumn* column;
196 GtkCellRenderer* renderer;
199 cfg = stats_tree_get_cfg_by_abbr(abbr);
201 if (cfg && cfg->in_use) {
203 report_failure("cannot open more than one tree of the same type at once");
208 init_strlen = strlen(cfg->pr->stat_dlg->init_string);
210 if (strncmp (optarg, cfg->pr->stat_dlg->init_string, init_strlen) == 0){
211 if (init_strlen == strlen(optarg)) {
212 st = stats_tree_new(cfg,pr,NULL);
214 st = stats_tree_new(cfg,pr,(char*)optarg+init_strlen+1);
218 st = stats_tree_new(cfg,pr,NULL);
221 report_failure("no such stats_tree (%s) in stats_tree registry",abbr);
228 report_failure("could not obtain stats_tree abbr from optarg");
235 window_name = g_strdup_printf("%s Stats Tree", cfg->name);
237 st->pr->win = window_new_with_geom(GTK_WINDOW_TOPLEVEL,window_name,window_name);
238 gtk_window_set_default_size(GTK_WINDOW(st->pr->win), 400, 400);
242 title=g_strdup_printf("%s with filter: %s",cfg->name,st->filter);
245 title=g_strdup_printf("%s", cfg->name);
248 gtk_window_set_title(GTK_WINDOW(st->pr->win), title);
251 main_vb = gtk_vbox_new(FALSE, 3);
252 gtk_container_set_border_width(GTK_CONTAINER(main_vb), 12);
253 gtk_container_add(GTK_CONTAINER(st->pr->win), main_vb);
255 scr_win = scrolled_window_new(NULL, NULL);
257 st->pr->store = gtk_tree_store_new (N_COLUMNS, G_TYPE_STRING, G_TYPE_STRING,
258 G_TYPE_STRING, G_TYPE_STRING);
260 st->pr->tree = gtk_tree_view_new_with_model (GTK_TREE_MODEL (st->pr->store));
261 g_object_unref(G_OBJECT(st->pr->store));
263 gtk_container_add( GTK_CONTAINER(scr_win), st->pr->tree);
266 renderer = gtk_cell_renderer_text_new ();
267 column = gtk_tree_view_column_new_with_attributes ("Topic / Item", renderer,
268 "text", TITLE_COLUMN,
270 gtk_tree_view_column_set_resizable (column,TRUE);
271 gtk_tree_view_column_set_sizing(column,GTK_TREE_VIEW_COLUMN_AUTOSIZE);
272 gtk_tree_view_append_column (GTK_TREE_VIEW (st->pr->tree), column);
274 renderer = gtk_cell_renderer_text_new ();
275 column = gtk_tree_view_column_new_with_attributes ("Count", renderer,
276 "text", COUNT_COLUMN,
279 gtk_tree_view_column_set_resizable (column,TRUE);
280 gtk_tree_view_column_set_sizing(column,GTK_TREE_VIEW_COLUMN_AUTOSIZE);
281 gtk_tree_view_append_column (GTK_TREE_VIEW (st->pr->tree), column);
283 renderer = gtk_cell_renderer_text_new ();
284 column = gtk_tree_view_column_new_with_attributes ("Rate", renderer,
287 gtk_tree_view_column_set_resizable (column,TRUE);
288 gtk_tree_view_column_set_sizing(column,GTK_TREE_VIEW_COLUMN_AUTOSIZE);
289 gtk_tree_view_append_column (GTK_TREE_VIEW (st->pr->tree), column);
291 renderer = gtk_cell_renderer_text_new ();
292 column = gtk_tree_view_column_new_with_attributes ("Percent", renderer,
293 "text", PERCENT_COLUMN,
295 gtk_tree_view_column_set_resizable(column,TRUE);
296 gtk_tree_view_column_set_sizing(column,GTK_TREE_VIEW_COLUMN_AUTOSIZE);
297 gtk_tree_view_append_column (GTK_TREE_VIEW (st->pr->tree), column);
299 gtk_container_add( GTK_CONTAINER(main_vb), scr_win);
301 error_string = register_tap_listener( cfg->tapname,
309 /* error, we failed to attach to the tap. clean up */
310 simple_dialog( ESD_TYPE_ERROR, ESD_BTN_OK, "%s", error_string->str );
311 /* destroy_stat_tree_window(st); */
312 report_failure("stats_tree for: %s failed to attach to the tap: %s",cfg->name,error_string->str);
313 g_string_free(error_string, TRUE);
317 bbox = dlg_button_row_new(GTK_STOCK_CLOSE, NULL);
318 gtk_box_pack_start(GTK_BOX(main_vb), bbox, FALSE, FALSE, 0);
320 bt_close = g_object_get_data(G_OBJECT(bbox), GTK_STOCK_CLOSE);
321 window_set_cancel_button(st->pr->win, bt_close, window_cancel_button_cb);
323 g_signal_connect(GTK_WINDOW(st->pr->win), "delete_event", G_CALLBACK(window_delete_event_cb), NULL);
324 g_signal_connect(GTK_WINDOW(st->pr->win), "destroy", G_CALLBACK(free_gtk_tree), st);
326 gtk_widget_show_all(st->pr->win);
327 window_present(st->pr->win);
329 cf_retap_packets(&cfile, FALSE);
330 gdk_window_raise(st->pr->win->window);
335 register_gtk_stats_tree_tap (gpointer k _U_, gpointer v, gpointer p _U_)
337 stats_tree_cfg* cfg = v;
339 cfg->pr = g_malloc(sizeof(tree_pres));
341 cfg->pr->stat_dlg = g_malloc(sizeof(tap_dfilter_dlg));
343 cfg->pr->stat_dlg->win_title = g_strdup_printf("%s Stats Tree",cfg->name);
344 cfg->pr->stat_dlg->init_string = g_strdup_printf("%s,tree",cfg->abbr);
345 cfg->pr->stat_dlg->tap_init_cb = init_gtk_tree;
346 cfg->pr->stat_dlg->index = -1;
348 register_dfilter_stat(cfg->pr->stat_dlg, cfg->name, cfg->stat_group);
352 free_tree_presentation(stats_tree* st)
358 register_tap_listener_stats_tree_stat(void)
361 stats_tree_presentation(register_gtk_stats_tree_tap,
367 free_tree_presentation,