/* graph_analysis.c * Graphic Analysis addition for ethereal * * $Id$ * * Copyright 2004, Verso Technologies Inc. * By Alejandro Vaquero * * based on rtp_analysis.c and io_stat * * * Ethereal - Network traffic analyzer * By Gerald Combs * 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 "graph_analysis.h" #include #include "util.h" #include #include "register.h" #include #include /* in /gtk ... */ #include #include #include "gtkglobals.h" #include "dlg_utils.h" #include "ui_util.h" #include "main.h" #include "compat_macros.h" #include "../color.h" #include "epan/filesystem.h" #include #ifdef HAVE_UNISTD_H #include #endif #include "simple_dialog.h" /****************************************************************************/ #define OK_TEXT "[ Ok ]" #define PT_UNDEFINED -1 #if GTK_MAJOR_VERSION < 2 GtkRcStyle *rc_style; GdkColormap *colormap; #endif static GtkWidget *save_to_file_w = NULL; /****************************************************************************/ /* Reset the user_data structure */ static void graph_analysis_reset(graph_analysis_data_t* user_data) { int i; user_data->num_nodes = 0; user_data->num_items = 0; for (i=0; inodes[i].type = AT_NONE; user_data->nodes[i].len = 0; g_free((void *)user_data->nodes[i].data); user_data->nodes[i].data = NULL; } user_data->dlg.first_node=0; user_data->dlg.first_item=0; user_data->dlg.left_x_border=0; user_data->dlg.selected_item=0xFFFFFFFF; /*not item selected */ } /****************************************************************************/ /* Reset the user_data structure */ static void graph_analysis_init_dlg(graph_analysis_data_t* user_data) { int i; user_data->num_nodes = 0; user_data->num_items = 0; for (i=0; inodes[i].type = AT_NONE; user_data->nodes[i].len = 0; user_data->nodes[i].data = NULL; } user_data->dlg.first_node=0; user_data->dlg.first_item=0; user_data->dlg.left_x_border=0; user_data->dlg.selected_item=0xFFFFFFFF; /*not item selected */ /* init dialog_graph */ user_data->dlg.needs_redraw=TRUE; user_data->dlg.draw_area=NULL; user_data->dlg.pixmap=NULL; user_data->dlg.draw_area_comments=NULL; user_data->dlg.pixmap_comments=NULL; user_data->dlg.h_scrollbar=NULL; user_data->dlg.h_scrollbar_adjustment=NULL; user_data->dlg.v_scrollbar=NULL; user_data->dlg.v_scrollbar_adjustment=NULL; user_data->dlg.pixmap_width=350; user_data->dlg.pixmap_height=400; user_data->dlg.first_node=0; user_data->dlg.first_item=0; user_data->dlg.left_x_border=0; user_data->dlg.selected_item=0xFFFFFFFF; /*not item selected */ user_data->dlg.window=NULL; } /****************************************************************************/ /* CALLBACKS */ /****************************************************************************/ /* close the dialog window and remove the tap listener */ static void on_destroy(GtkWidget *win _U_, graph_analysis_data_t *user_data _U_) { int i; for (i=0; inodes[i].type = AT_NONE; user_data->nodes[i].len = 0; g_free((void *)user_data->nodes[i].data); user_data->nodes[i].data = NULL; } user_data->dlg.window = NULL; } /****************************************************************************/ #if 0 static void dialog_graph_set_title(graph_analysis_data_t* user_data) { char *title; if (!user_data->dlg.window){ return; } title = g_strdup_printf("Ale"); gtk_window_set_title(GTK_WINDOW(user_data->dlg.window), title); g_free(title); } #endif #define RIGHT_ARROW 1 #define LEFT_ARROW 0 #define WIDTH_ARROW 8 #define HEIGHT_ARROW 6 /****************************************************************************/ static void draw_arrow(GdkDrawable *pixmap, GdkGC *gc, gint x, gint y, gboolean direction) { GdkPoint arrow_point[3]; arrow_point[0].x = x; arrow_point[0].y = y-HEIGHT_ARROW/2; if (direction == RIGHT_ARROW) arrow_point[1].x = x+WIDTH_ARROW; else arrow_point[1].x = x-WIDTH_ARROW; arrow_point[1].y = y; arrow_point[2].x = x; arrow_point[2].y = y+HEIGHT_ARROW/2;; gdk_draw_polygon(pixmap, gc, TRUE, arrow_point, 3); } #define MAX_LABEL 50 #define MAX_COMMENT 100 #define ITEM_HEIGHT 20 #define NODE_WIDTH 100 #define TOP_Y_BORDER 40 #define BOTTOM_Y_BORDER 0 #define COMMENT_WIDTH 400 #define NODE_CHARS_WIDTH 20 #define CONV_TIME_HEADER "Conv.| Time " #define EMPTY_HEADER " | " #define HEADER_LENGTH 16 /****************************************************************************/ /* adds trailing characters to complete the requested length */ /* NB: does not allocate new memory for the string, there must be enough */ /****************************************************************************/ static void enlarge_string(char *string, guint32 length, char pad){ guint32 i,l; l = strlen(string); if (l>=length){ return; } for (i=l;i p2){ first = p2; last = p1; } else{ first = p1; last = p2; } if ((unsigned int)(last - first)>strlen(text_to_insert)){ last = first + strlen(text_to_insert); } for (i=first;idlg.save_file,"w"); if (of==NULL){ return FALSE; } first_item = user_data->dlg.first_item; /* get the items to display and fill the matrix array */ list = g_list_first(user_data->graph_info->list); current_item = 0; i = 0; while (list) { gai = list->data; if (gai->display){ if (i>=first_item){ user_data->dlg.items[current_item].frame_num = gai->frame_num; user_data->dlg.items[current_item].time = gai->time; user_data->dlg.items[current_item].port_src = gai->port_src; user_data->dlg.items[current_item].port_dst = gai->port_dst; user_data->dlg.items[current_item].frame_label = gai->frame_label; user_data->dlg.items[current_item].comment = gai->comment; user_data->dlg.items[current_item].conv_num = gai->conv_num; user_data->dlg.items[current_item].src_node = gai->src_node; user_data->dlg.items[current_item].dst_node = gai->dst_node; current_item++; } i++; } list = g_list_next(list); } display_items = current_item; /* if not items to display */ if (display_items == 0) return TRUE; display_nodes=user_data->num_nodes; first_node = user_data->dlg.first_node; /* Write the conv. and time headers */ fprintf (of, CONV_TIME_HEADER); empty_line = g_strdup(""); /* Write the node names on top */ for (i=0; inodes[i+first_node]))); enlarge_string(label_string,NODE_CHARS_WIDTH,' '); fprintf(of,label_string); strcpy(label_string,"| "); enlarge_string(label_string,NODE_CHARS_WIDTH,' '); tmp_str = g_strdup(empty_line); g_free(empty_line); empty_line = g_strdup_printf("%s%s",tmp_str,label_string); g_free(tmp_str); } tmp_str = g_strdup(empty_line); g_free(empty_line); empty_line = g_strdup_printf("%s|",tmp_str); g_free(tmp_str); separator_line = g_malloc(strlen(empty_line)+HEADER_LENGTH+1); separator_line[0]='\0'; enlarge_string(separator_line,strlen(empty_line)+HEADER_LENGTH,'-'); separator_line[strlen(separator_line)-1]='\n'; fprintf(of,"|\n"); /* * Draw the items */ for (current_item=0; current_itemdlg.items[current_item].src_node-first_node)*NODE_CHARS_WIDTH+NODE_CHARS_WIDTH/2; end_position = (user_data->dlg.items[current_item].dst_node-first_node)*NODE_CHARS_WIDTH+NODE_CHARS_WIDTH/2; if (start_position > end_position){ item_width=start_position-end_position; } else if (start_position < end_position){ item_width=end_position-start_position; } else{ /* same origin and destination address */ end_position = start_position+NODE_CHARS_WIDTH; item_width = NODE_CHARS_WIDTH; } /* separator between conversations */ if (user_data->dlg.items[current_item].conv_num != old_conv_num){ fprintf(of,separator_line); old_conv_num=user_data->dlg.items[current_item].conv_num; } /* write the conversation number */ g_snprintf(label_string, 5, "%i", user_data->dlg.items[current_item].conv_num); enlarge_string(label_string,5,' '); fprintf(of,"%s",label_string); /* write the time */ g_snprintf(label_string, 11, "|%.3f", user_data->dlg.items[current_item].time); enlarge_string(label_string,11,' '); fprintf(of,"%s",label_string); /* write the frame label */ tmp_str = g_strdup(empty_line); overwrite(tmp_str,user_data->dlg.items[current_item].frame_label, start_position, end_position ); fprintf(of,tmp_str); /* write the comments */ g_snprintf(label_string, MAX_COMMENT, "%s", user_data->dlg.items[current_item].comment); fprintf(of,"%s\n",label_string); /* write draw the arrow and frame label*/ fprintf(of,EMPTY_HEADER); tmp_str = g_strdup(empty_line); tmp_str2 = g_malloc(item_width); tmp_str2[0]='\0'; enlarge_string(tmp_str2,item_width-1,'-'); if (start_positiondlg.items[current_item].port_src); g_snprintf(dst_port,7,"(%i)", user_data->dlg.items[current_item].port_dst); if (start_positiondlg.save_file = g_strdup(gtk_file_selection_get_filename(GTK_FILE_SELECTION (save_to_file_w))); /* Perhaps the user specified a directory instead of a file. Check whether they did. */ if (test_for_directory(user_data_p->dlg.save_file) == EISDIR) { /* It's a directory - set the file selection box to display it. */ set_last_open_dir(user_data_p->dlg.save_file); g_free(user_data_p->dlg.save_file); file_selection_set_current_folder(save_to_file_w, get_last_open_dir()); return; } /* check whether the file exists */ file_test = fopen(user_data_p->dlg.save_file,"r"); if (file_test!=NULL){ dialog = simple_dialog(ESD_TYPE_CONFIRMATION, ESD_BTNS_YES_NO, "%sFile: \"%s\" already exists!%s\n\n" "Do you want to overwrite it?", simple_dialog_primary_start(),user_data_p->dlg.save_file, simple_dialog_primary_end()); simple_dialog_set_cb(dialog, overwrite_existing_file_cb, user_data); fclose(file_test); } else{ if (!dialog_graph_dump_to_file(user_data)) return; } window_destroy(GTK_WIDGET(save_to_file_w)); } /****************************************************************************/ static void on_save_bt_clicked (GtkButton *button _U_, gpointer user_data _U_) { GtkWidget *vertb; GtkWidget *ok_bt; if (save_to_file_w != NULL) { /* There's already a Save to file dialog box; reactivate it. */ reactivate_window(save_to_file_w); return; } save_to_file_w = gtk_file_selection_new("Ethereal: Save graph to file"); /* Container for each row of widgets */ vertb = gtk_vbox_new(FALSE, 0); gtk_container_border_width(GTK_CONTAINER(vertb), 5); gtk_box_pack_start(GTK_BOX(GTK_FILE_SELECTION(save_to_file_w)->action_area), vertb, FALSE, FALSE, 0); gtk_widget_show (vertb); ok_bt = GTK_FILE_SELECTION(save_to_file_w)->ok_button; SIGNAL_CONNECT(ok_bt, "clicked", save_to_file_ok_cb, user_data); window_set_cancel_button(save_to_file_w, GTK_FILE_SELECTION(save_to_file_w)->cancel_button, window_cancel_button_cb); SIGNAL_CONNECT(save_to_file_w, "delete_event", window_delete_event_cb, NULL); SIGNAL_CONNECT(save_to_file_w, "destroy", save_to_file_destroy_cb, NULL); gtk_widget_show(save_to_file_w); window_present(save_to_file_w); } /****************************************************************************/ static void dialog_graph_draw(graph_analysis_data_t* user_data) { guint32 i, last_item, first_item, last_node, first_node, display_items, display_nodes; guint32 start_arrow, end_arrow, label_x, src_port_x, dst_port_x, arrow_width; guint32 current_item; guint32 left_x_border; guint32 right_x_border; guint32 top_y_border; guint32 bottom_y_border; graph_analysis_item_t *gai; gboolean display_label; #if GTK_MAJOR_VERSION < 2 GdkFont *font; FONT_TYPE *big_font; FONT_TYPE *small_font; #else PangoLayout *layout; PangoLayout *big_layout; PangoLayout *small_layout; #endif guint32 label_width, label_height; guint32 draw_width, draw_height; char label_string[MAX_COMMENT]; GList* list; /* new variables */ #if GTK_MAJOR_VERSION <2 font = user_data->dlg.draw_area->style->font; big_font = gdk_font_load("-adobe-helvetica-bold-r-normal--12-120-75-75-p-70-iso8859-1"); small_font = gdk_font_load("-adobe-helvetica-bold-r-normal--10-120-75-75-p-70-iso8859-1"); #endif if(!user_data->dlg.needs_redraw){ return; } user_data->dlg.needs_redraw=FALSE; /* * Clear out old plot */ gdk_draw_rectangle(user_data->dlg.pixmap, user_data->dlg.draw_area->style->white_gc, TRUE, 0, 0, user_data->dlg.draw_area->allocation.width, user_data->dlg.draw_area->allocation.height); gdk_draw_rectangle(user_data->dlg.pixmap_comments, user_data->dlg.draw_area->style->white_gc, TRUE, 0, 0, COMMENT_WIDTH, user_data->dlg.draw_area->allocation.height); /* Calculate the y border */ top_y_border=TOP_Y_BORDER; /* to display the node address */ bottom_y_border=2; draw_height=user_data->dlg.pixmap_height-top_y_border-bottom_y_border; first_item = user_data->dlg.first_item; display_items = draw_height/ITEM_HEIGHT; last_item = first_item+display_items-1; /* get the items to display and fill the matrix array */ list = g_list_first(user_data->graph_info->list); current_item = 0; i = 0; while (list) { gai = list->data; if (gai->display){ if (current_item>=display_items) break; /* the item is outside the display */ if (i>=first_item){ user_data->dlg.items[current_item].frame_num = gai->frame_num; user_data->dlg.items[current_item].time = gai->time; user_data->dlg.items[current_item].port_src = gai->port_src; user_data->dlg.items[current_item].port_dst = gai->port_dst; /* Add "..." if the length is 50 characters */ if (strlen(gai->frame_label) > 48) { gai->frame_label[48] = '.'; gai->frame_label[47] = '.'; gai->frame_label[46] = '.'; } user_data->dlg.items[current_item].frame_label = gai->frame_label; user_data->dlg.items[current_item].comment = gai->comment; user_data->dlg.items[current_item].conv_num = gai->conv_num; user_data->dlg.items[current_item].src_node = gai->src_node; user_data->dlg.items[current_item].dst_node = gai->dst_node; user_data->dlg.items[current_item].line_style = gai->line_style; current_item++; } i++; } list = g_list_next(list); } /* in case the windows is resized so we have to move the top item */ if ((first_item + display_items) > user_data->num_items){ if (display_items>user_data->num_items) first_item=0; else first_item = user_data->num_items - display_items; } /* in case there are less items than possible displayed */ display_items = current_item; last_item = first_item+display_items-1; /* if not items to display */ if (display_items == 0) return; /* Calculate the x borders */ /* We use time from the last display item to calcultate the x left border */ g_snprintf(label_string, MAX_LABEL, "%.3f", user_data->dlg.items[display_items-1].time); #if GTK_MAJOR_VERSION < 2 label_width=gdk_string_width(font, label_string); label_height=gdk_string_height(font, label_string); #else layout = gtk_widget_create_pango_layout(user_data->dlg.draw_area, label_string); big_layout = gtk_widget_create_pango_layout(user_data->dlg.draw_area, label_string); small_layout = gtk_widget_create_pango_layout(user_data->dlg.draw_area, label_string); /* XXX - to prevent messages like "Couldn't load font x, falling back to y", I've changed font description from "Helvetica-Bold 8" to "Helvetica,Sans,Bold 8", this seems to be conforming to the API, see http://developer.gnome.org/doc/API/2.0/pango/pango-Fonts.html */ pango_layout_set_font_description(big_layout, pango_font_description_from_string("Helvetica,Sans,Bold 8")); pango_layout_set_font_description(small_layout, pango_font_description_from_string("Helvetica,Sans,Bold 7")); pango_layout_get_pixel_size(layout, &label_width, &label_height); #endif left_x_border=label_width+10; user_data->dlg.left_x_border = left_x_border; right_x_border=2; /* Calculate the number of nodes to display */ draw_width=user_data->dlg.pixmap_width-right_x_border-left_x_border; display_nodes = draw_width/NODE_WIDTH; first_node = user_data->dlg.first_node; /* in case the windows is resized so we have to move the left node */ if ((first_node + display_nodes) > user_data->num_nodes){ if (display_nodes>user_data->num_nodes) first_node=0; else first_node=user_data->num_nodes - display_nodes; } /* in case there are less nodes than possible displayed */ if (display_nodes>user_data->num_nodes) display_nodes=user_data->num_nodes; last_node = first_node + display_nodes-1; /* Paint the background items */ for (current_item=0; current_itemdlg.pixmap, user_data->dlg.bg_gc[user_data->dlg.items[current_item].conv_num%MAX_NUM_COL_CONV], TRUE, left_x_border, top_y_border+current_item*ITEM_HEIGHT, draw_width, ITEM_HEIGHT); } /* Draw the node names on top and the division lines */ for (i=0; inodes[i+first_node]))); #if GTK_MAJOR_VERSION < 2 label_width=gdk_string_width(font, label_string); label_height=gdk_string_height(font, label_string); gdk_draw_string(user_data->dlg.pixmap, font, user_data->dlg.draw_area->style->black_gc, left_x_border+NODE_WIDTH/2-label_width/2+NODE_WIDTH*i, top_y_border/2-label_height/2, label_string); #else pango_layout_set_text(layout, label_string, -1); pango_layout_get_pixel_size(layout, &label_width, &label_height); gdk_draw_layout(user_data->dlg.pixmap, user_data->dlg.draw_area->style->black_gc, left_x_border+NODE_WIDTH/2-label_width/2+NODE_WIDTH*i, top_y_border/2-label_height/2, layout); #endif /* draw the node division lines */ gdk_draw_line(user_data->dlg.pixmap, user_data->dlg.div_line_gc, left_x_border+NODE_WIDTH/2+NODE_WIDTH*i, top_y_border, left_x_border+NODE_WIDTH/2+NODE_WIDTH*i, user_data->dlg.pixmap_height-bottom_y_border); } /* * Draw the items */ for (current_item=0; current_itemdlg.items[current_item].time); #if GTK_MAJOR_VERSION < 2 label_width=gdk_string_width(font, label_string); label_height=gdk_string_height(font, label_string); gdk_draw_string(user_data->dlg.pixmap, font, user_data->dlg.draw_area->style->black_gc, left_x_border-label_width-4, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT/2+label_height/4, label_string); #else pango_layout_set_text(layout, label_string, -1); pango_layout_get_pixel_size(layout, &label_width, &label_height); gdk_draw_layout(user_data->dlg.pixmap, user_data->dlg.draw_area->style->black_gc, left_x_border-label_width-4, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT/2-label_height/2, layout); #endif /*draw the comments */ g_snprintf(label_string, MAX_COMMENT, "%s", user_data->dlg.items[current_item].comment); #if GTK_MAJOR_VERSION < 2 label_width=gdk_string_width(small_font, label_string); label_height=gdk_string_height(small_font, label_string); gdk_draw_string(user_data->dlg.pixmap_comments, small_font, user_data->dlg.draw_area_comments->style->black_gc, 2, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT/2+label_height/4, label_string); #else pango_layout_set_text(small_layout, label_string, -1); pango_layout_get_pixel_size(small_layout, &label_width, &label_height); gdk_draw_layout(user_data->dlg.pixmap_comments, user_data->dlg.draw_area->style->black_gc, 2, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT/2-label_height/2, small_layout); #endif /* draw the arrow an frame label*/ display_label = FALSE; if (user_data->dlg.items[current_item].src_node>=first_node){ if (user_data->dlg.items[current_item].src_node<=last_node){ start_arrow = left_x_border+(user_data->dlg.items[current_item].src_node-first_node)*NODE_WIDTH+NODE_WIDTH/2; display_label = TRUE; } else { start_arrow = user_data->dlg.pixmap_width - right_x_border; } } else { start_arrow = left_x_border; } if (user_data->dlg.items[current_item].dst_node>=first_node){ if (user_data->dlg.items[current_item].dst_node<=last_node){ end_arrow = left_x_border+(user_data->dlg.items[current_item].dst_node-first_node)*NODE_WIDTH+NODE_WIDTH/2; display_label = TRUE; } else { end_arrow = user_data->dlg.pixmap_width - right_x_border; } } else { end_arrow = left_x_border; } if (start_arrow != end_arrow){ /* draw the arrow line */ gdk_draw_line(user_data->dlg.pixmap, user_data->dlg.draw_area->style->black_gc, start_arrow, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-7, end_arrow, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-7); /* draw the additional line when line style is 2 pixels width */ if (user_data->dlg.items[current_item].line_style == 2){ gdk_draw_line(user_data->dlg.pixmap, user_data->dlg.draw_area->style->black_gc, start_arrow, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-6, end_arrow, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-6); } /* draw the arrow */ if (start_arrowdlg.pixmap, user_data->dlg.draw_area->style->black_gc, end_arrow-WIDTH_ARROW,top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-7, RIGHT_ARROW); else draw_arrow(user_data->dlg.pixmap, user_data->dlg.draw_area->style->black_gc, end_arrow+WIDTH_ARROW,top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-7, LEFT_ARROW); } /* draw the frame comment */ if (display_label){ g_snprintf(label_string, MAX_LABEL, "%s", user_data->dlg.items[current_item].frame_label); #if GTK_MAJOR_VERSION < 2 label_width=gdk_string_width(big_font, label_string); label_height=gdk_string_height(big_font, label_string); #else pango_layout_set_text(big_layout, label_string, -1); pango_layout_get_pixel_size(big_layout, &label_width, &label_height); #endif if (start_arrowarrow_width) arrow_width = label_width; if (left_x_border > (label_x-label_width/2)) label_x = left_x_border + label_width/2; if ((user_data->dlg.pixmap_width - right_x_border) < (label_x+label_width/2)) label_x = user_data->dlg.pixmap_width - right_x_border - label_width/2; #if GTK_MAJOR_VERSION < 2 gdk_draw_string(user_data->dlg.pixmap, big_font, user_data->dlg.draw_area->style->black_gc, label_x - label_width/2, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT/2+label_height/4-3, label_string); #else gdk_draw_layout(user_data->dlg.pixmap, user_data->dlg.draw_area->style->black_gc, label_x - label_width/2, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT/2-label_height/2-3, big_layout); #endif /* draw the source port number */ if ((start_arrow != left_x_border) && (start_arrow != (user_data->dlg.pixmap_width - right_x_border))){ g_snprintf(label_string, MAX_LABEL, "(%i)", user_data->dlg.items[current_item].port_src); #if GTK_MAJOR_VERSION < 2 label_width=gdk_string_width(small_font, label_string); label_height=gdk_string_height(small_font, label_string); #else pango_layout_set_text(small_layout, label_string, -1); pango_layout_get_pixel_size(small_layout, &label_width, &label_height); #endif if (start_arrowdlg.pixmap, small_font, user_data->dlg.div_line_gc, src_port_x, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-2+label_height/4-2, label_string); #else gdk_draw_layout(user_data->dlg.pixmap, user_data->dlg.div_line_gc, src_port_x, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-2-label_height/2-2, small_layout); #endif } /* draw the destination port number */ if ((end_arrow != left_x_border) && (end_arrow != (user_data->dlg.pixmap_width - right_x_border))){ g_snprintf(label_string, MAX_LABEL, "(%i)", user_data->dlg.items[current_item].port_dst); #if GTK_MAJOR_VERSION < 2 label_width=gdk_string_width(small_font, label_string); label_height=gdk_string_height(small_font, label_string); #else pango_layout_set_text(small_layout, label_string, -1); pango_layout_get_pixel_size(small_layout, &label_width, &label_height); #endif if (start_arrowdlg.pixmap, small_font, user_data->dlg.div_line_gc, dst_port_x, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-2+label_height/4-2, label_string); #else gdk_draw_layout(user_data->dlg.pixmap, user_data->dlg.div_line_gc, dst_port_x, top_y_border+current_item*ITEM_HEIGHT+ITEM_HEIGHT-2-label_height/2-2, small_layout); #endif } } } #if GTK_MAJOR_VERSION >= 2 g_object_unref(G_OBJECT(layout)); #endif /* draw the border on the selected item */ if ( (user_data->dlg.selected_item != 0xFFFFFFFF) && ( (user_data->dlg.selected_item>=first_item) && (user_data->dlg.selected_item<=last_item) )){ gdk_draw_rectangle(user_data->dlg.pixmap, user_data->dlg.draw_area->style->black_gc, FALSE, left_x_border-1, (user_data->dlg.selected_item-first_item)*ITEM_HEIGHT+TOP_Y_BORDER, user_data->dlg.pixmap_width-left_x_border-right_x_border+1, ITEM_HEIGHT); } /* refresh the draw areas */ gdk_draw_pixmap(user_data->dlg.draw_area->window, user_data->dlg.draw_area->style->fg_gc[GTK_WIDGET_STATE(user_data->dlg.draw_area)], user_data->dlg.pixmap, 0, 0, 0, 0, user_data->dlg.pixmap_width, user_data->dlg.pixmap_height); gdk_draw_pixmap(user_data->dlg.draw_area_comments->window, user_data->dlg.draw_area_comments->style->fg_gc[GTK_WIDGET_STATE(user_data->dlg.draw_area_comments)], user_data->dlg.pixmap_comments, 0, 0, 0, 0, COMMENT_WIDTH, user_data->dlg.pixmap_height); /* update the h_scrollbar */ user_data->dlg.h_scrollbar_adjustment->upper=(gfloat) user_data->num_nodes-1; user_data->dlg.h_scrollbar_adjustment->step_increment=1; user_data->dlg.h_scrollbar_adjustment->page_increment=(gfloat) (last_node-first_node); user_data->dlg.h_scrollbar_adjustment->page_size=(gfloat) (last_node-first_node); user_data->dlg.h_scrollbar_adjustment->value=(gfloat) first_node; gtk_adjustment_changed(user_data->dlg.h_scrollbar_adjustment); gtk_adjustment_value_changed(user_data->dlg.h_scrollbar_adjustment); /* update the v_scrollbar */ user_data->dlg.v_scrollbar_adjustment->upper=(gfloat) user_data->num_items-1; user_data->dlg.v_scrollbar_adjustment->step_increment=1; user_data->dlg.v_scrollbar_adjustment->page_increment=(gfloat) (last_item-first_item); user_data->dlg.v_scrollbar_adjustment->page_size=(gfloat) (last_item-first_item); user_data->dlg.v_scrollbar_adjustment->value=(gfloat) first_item; gtk_adjustment_changed(user_data->dlg.v_scrollbar_adjustment); gtk_adjustment_value_changed(user_data->dlg.v_scrollbar_adjustment); } /****************************************************************************/ static void dialog_graph_redraw(graph_analysis_data_t* user_data) { user_data->dlg.needs_redraw=TRUE; dialog_graph_draw(user_data); } /****************************************************************************/ static gint button_press_event(GtkWidget *widget, GdkEventButton *event _U_) { graph_analysis_data_t *user_data; guint32 item; user_data=(graph_analysis_data_t *)OBJECT_GET_DATA(widget, "graph_analysis_data_t"); if (event->type != GDK_BUTTON_PRESS) return TRUE; if (event->yy - TOP_Y_BORDER) / ITEM_HEIGHT; user_data->dlg.selected_item = item + user_data->dlg.first_item; user_data->dlg.needs_redraw=TRUE; dialog_graph_draw(user_data); cf_goto_frame(&cfile, user_data->dlg.items[item].frame_num); return TRUE; } #if GTK_MAJOR_VERSION >= 2 /* scroll events are not available in gtk-1.2 */ /****************************************************************************/ static gint scroll_event(GtkWidget *widget, GdkEventScroll *event) { graph_analysis_data_t *user_data; user_data=(graph_analysis_data_t *)OBJECT_GET_DATA(widget, "graph_analysis_data_t"); /* Up scroll */ switch(event->direction) { case(GDK_SCROLL_UP): if (user_data->dlg.first_item == 0) return TRUE; if (user_data->dlg.first_item < 3) user_data->dlg.first_item = 0; else user_data->dlg.first_item -= 3; break; case(GDK_SCROLL_DOWN): if ((user_data->dlg.first_item+user_data->dlg.v_scrollbar_adjustment->page_size+1 == user_data->num_items)) return TRUE; if ((user_data->dlg.first_item+user_data->dlg.v_scrollbar_adjustment->page_size+1) > (user_data->num_items-3)) user_data->dlg.first_item = user_data->num_items-(guint32)user_data->dlg.v_scrollbar_adjustment->page_size-1; else user_data->dlg.first_item += 3; break; case(GDK_SCROLL_LEFT): case(GDK_SCROLL_RIGHT): /* nothing to do */ break; } dialog_graph_redraw(user_data); return TRUE; } #endif /****************************************************************************/ static gint key_press_event(GtkWidget *widget, GdkEventKey *event _U_) { graph_analysis_data_t *user_data; user_data=(graph_analysis_data_t *)OBJECT_GET_DATA(widget, "graph_analysis_data_t"); /* if there is nothing selected, just return */ if (user_data->dlg.selected_item == 0xFFFFFFFF) return TRUE; /* Up arrow */ if (event->keyval == GDK_Up){ if (user_data->dlg.selected_item == 0) return TRUE; user_data->dlg.selected_item--; if ( (user_data->dlg.selected_itemdlg.first_item) || (user_data->dlg.selected_item>user_data->dlg.first_item+user_data->dlg.v_scrollbar_adjustment->page_size) ) user_data->dlg.first_item = user_data->dlg.selected_item; /* Down arrow */ } else if (event->keyval == GDK_Down){ if (user_data->dlg.selected_item == user_data->num_items-1) return TRUE; user_data->dlg.selected_item++; if ( (user_data->dlg.selected_itemdlg.first_item) || (user_data->dlg.selected_item>user_data->dlg.first_item+user_data->dlg.v_scrollbar_adjustment->page_size) ) user_data->dlg.first_item = (guint32)user_data->dlg.selected_item-(guint32)user_data->dlg.v_scrollbar_adjustment->page_size; } else if (event->keyval == GDK_Left){ if (user_data->dlg.first_node == 0) return TRUE; user_data->dlg.first_node--; } else if (event->keyval == GDK_Right){ if ((user_data->dlg.first_node+user_data->dlg.h_scrollbar_adjustment->page_size+1 == user_data->num_nodes)) return TRUE; user_data->dlg.first_node++; } else return TRUE; user_data->dlg.needs_redraw=TRUE; dialog_graph_draw(user_data); cf_goto_frame(&cfile, user_data->dlg.items[user_data->dlg.selected_item-user_data->dlg.first_item].frame_num); return TRUE; } /****************************************************************************/ static gint expose_event(GtkWidget *widget, GdkEventExpose *event) { graph_analysis_data_t *user_data; user_data=(graph_analysis_data_t *)OBJECT_GET_DATA(widget, "graph_analysis_data_t"); if(!user_data){ exit(10); } gdk_draw_pixmap(widget->window, widget->style->fg_gc[GTK_WIDGET_STATE(widget)], user_data->dlg.pixmap, event->area.x, event->area.y, event->area.x, event->area.y, event->area.width, event->area.height); return FALSE; } /****************************************************************************/ static gint expose_event_comments(GtkWidget *widget, GdkEventExpose *event) { graph_analysis_data_t *user_data; user_data=(graph_analysis_data_t *)OBJECT_GET_DATA(widget, "graph_analysis_data_t"); if(!user_data){ exit(10); } gdk_draw_pixmap(widget->window, widget->style->fg_gc[GTK_WIDGET_STATE(widget)], user_data->dlg.pixmap_comments, event->area.x, event->area.y, event->area.x, event->area.y, event->area.width, event->area.height); return FALSE; } static const GdkColor COLOR_GRAY = {0, 0x7fff, 0x7fff, 0x7fff}; /****************************************************************************/ static gint configure_event(GtkWidget *widget, GdkEventConfigure *event _U_) { graph_analysis_data_t *user_data; int i; GdkColor color_div_line = COLOR_GRAY; static GdkColor col[MAX_NUM_COL_CONV] = { {0, 0x00FF, 0xFFFF, 0x00FF}, {0, 0xFFFF, 0xFFFF, 0x00FF}, {0, 0xFFFF, 0x00FF, 0x00FF}, {0, 0xFFFF, 0x00FF, 0xFFFF}, {0, 0x00FF, 0x00FF, 0xFFFF}, {0, 0x00FF, 0xFFFF, 0xFFFF}, {0, 0xFFFF, 0x80FF, 0x00FF}, {0, 0x80FF, 0x00FF, 0xFFFF}, {0, 0x00FF, 0x80FF, 0xFFFF}, {0, 0xFFFF, 0x00FF, 0x80FF} }; user_data=(graph_analysis_data_t *)OBJECT_GET_DATA(widget, "graph_analysis_data_t"); if(!user_data){ exit(10); } if(user_data->dlg.pixmap){ gdk_pixmap_unref(user_data->dlg.pixmap); user_data->dlg.pixmap=NULL; } user_data->dlg.pixmap=gdk_pixmap_new(widget->window, widget->allocation.width, widget->allocation.height, -1); user_data->dlg.pixmap_width=widget->allocation.width; user_data->dlg.pixmap_height=widget->allocation.height; gdk_draw_rectangle(user_data->dlg.pixmap, widget->style->white_gc, TRUE, 0, 0, widget->allocation.width, widget->allocation.height); /* create gc for division lines and set the line stype to dash*/ user_data->dlg.div_line_gc=gdk_gc_new(user_data->dlg.pixmap); gdk_gc_set_line_attributes(user_data->dlg.div_line_gc, 1, GDK_LINE_ON_OFF_DASH, 0, 0); #if GTK_MAJOR_VERSION < 2 colormap = gtk_widget_get_colormap (widget); if (!gdk_color_alloc (colormap, &color_div_line)){ g_warning ("Couldn't allocate color"); } gdk_gc_set_foreground(user_data->dlg.div_line_gc, &color_div_line); #else gdk_gc_set_rgb_fg_color(user_data->dlg.div_line_gc, &color_div_line); #endif /* create gcs for the background items */ for (i=0; idlg.bg_gc[i]=gdk_gc_new(user_data->dlg.pixmap); #if GTK_MAJOR_VERSION < 2 colormap = gtk_widget_get_colormap (widget); if (!gdk_color_alloc (colormap, &col[i])){ g_warning ("Couldn't allocate color"); } gdk_gc_set_foreground(user_data->dlg.bg_gc[i], &col[i]); #else gdk_gc_set_rgb_fg_color(user_data->dlg.bg_gc[i], &col[i]); #endif } dialog_graph_redraw(user_data); return TRUE; } /****************************************************************************/ static gint configure_event_comments(GtkWidget *widget, GdkEventConfigure *event _U_) { graph_analysis_data_t *user_data; user_data=(graph_analysis_data_t *)OBJECT_GET_DATA(widget, "graph_analysis_data_t"); if(!user_data){ exit(10); } if(user_data->dlg.pixmap_comments){ gdk_pixmap_unref(user_data->dlg.pixmap_comments); user_data->dlg.pixmap_comments=NULL; } user_data->dlg.pixmap_comments=gdk_pixmap_new(widget->window, COMMENT_WIDTH, widget->allocation.height, -1); gdk_draw_rectangle(user_data->dlg.pixmap_comments, widget->style->white_gc, TRUE, 0, 0, COMMENT_WIDTH, widget->allocation.height); dialog_graph_redraw(user_data); return TRUE; } /****************************************************************************/ static gint h_scrollbar_changed(GtkWidget *widget _U_, gpointer data) { graph_analysis_data_t *user_data=(graph_analysis_data_t *)data; if ((user_data->dlg.first_node+user_data->dlg.h_scrollbar_adjustment->page_size+1 == user_data->num_nodes) && (user_data->dlg.h_scrollbar_adjustment->value >= user_data->dlg.first_node )) return TRUE; if (user_data->dlg.first_node == (guint16) user_data->dlg.h_scrollbar_adjustment->value) return TRUE; user_data->dlg.first_node = (guint16) user_data->dlg.h_scrollbar_adjustment->value; dialog_graph_redraw(user_data); return TRUE; } /****************************************************************************/ static gint v_scrollbar_changed(GtkWidget *widget _U_, gpointer data) { graph_analysis_data_t *user_data=(graph_analysis_data_t *)data; if ((user_data->dlg.first_item+user_data->dlg.v_scrollbar_adjustment->page_size+1 == user_data->num_items) && (user_data->dlg.v_scrollbar_adjustment->value >= user_data->dlg.first_item )) return TRUE; if (user_data->dlg.first_item == user_data->dlg.v_scrollbar_adjustment->value) return TRUE; user_data->dlg.first_item = (guint32) user_data->dlg.v_scrollbar_adjustment->value; dialog_graph_redraw(user_data); return TRUE; } /****************************************************************************/ static void create_draw_area(graph_analysis_data_t* user_data, GtkWidget *box) { GtkWidget *vbox; GtkWidget *hbox; GtkWidget *scroll_window; GtkWidget *viewport; hbox=gtk_hbox_new(FALSE, 0); gtk_widget_show(hbox); vbox=gtk_vbox_new(FALSE, 0); gtk_widget_show(vbox); /* create "comments" draw area */ user_data->dlg.draw_area_comments=gtk_drawing_area_new(); WIDGET_SET_SIZE(user_data->dlg.draw_area_comments, COMMENT_WIDTH, user_data->dlg.pixmap_height); scroll_window=gtk_scrolled_window_new(NULL, NULL); WIDGET_SET_SIZE(scroll_window, COMMENT_WIDTH/2, user_data->dlg.pixmap_height); gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scroll_window), GTK_POLICY_ALWAYS, GTK_POLICY_NEVER); viewport = gtk_viewport_new(gtk_scrolled_window_get_hadjustment(GTK_SCROLLED_WINDOW(scroll_window)), gtk_scrolled_window_get_vadjustment(GTK_SCROLLED_WINDOW(scroll_window))); gtk_container_add(GTK_CONTAINER(viewport), user_data->dlg.draw_area_comments); gtk_container_add(GTK_CONTAINER(scroll_window), viewport); gtk_viewport_set_shadow_type(GTK_VIEWPORT(viewport), GTK_SHADOW_NONE); OBJECT_SET_DATA(user_data->dlg.draw_area_comments, "graph_analysis_data_t", user_data); gtk_widget_add_events (user_data->dlg.draw_area_comments, GDK_BUTTON_PRESS_MASK); #if GTK_MAJOR_VERSION >= 2 SIGNAL_CONNECT(user_data->dlg.draw_area_comments, "scroll_event", scroll_event, user_data); #endif /* create main Graph draw area */ user_data->dlg.draw_area=gtk_drawing_area_new(); GTK_WIDGET_SET_FLAGS(user_data->dlg.draw_area, GTK_CAN_FOCUS); gtk_widget_grab_focus(user_data->dlg.draw_area); OBJECT_SET_DATA(user_data->dlg.draw_area, "graph_analysis_data_t", user_data); WIDGET_SET_SIZE(user_data->dlg.draw_area, user_data->dlg.pixmap_width, user_data->dlg.pixmap_height); /* signals needed to handle backing pixmap */ SIGNAL_CONNECT(user_data->dlg.draw_area, "expose_event", expose_event, NULL); SIGNAL_CONNECT(user_data->dlg.draw_area, "configure_event", configure_event, user_data); /* signals needed to handle backing pixmap comments*/ SIGNAL_CONNECT(user_data->dlg.draw_area_comments, "expose_event", expose_event_comments, NULL); SIGNAL_CONNECT(user_data->dlg.draw_area_comments, "configure_event", configure_event_comments, user_data); gtk_widget_add_events (user_data->dlg.draw_area, GDK_BUTTON_PRESS_MASK); SIGNAL_CONNECT(user_data->dlg.draw_area, "button_press_event", button_press_event, user_data); #if GTK_MAJOR_VERSION >= 2 SIGNAL_CONNECT(user_data->dlg.draw_area, "scroll_event", scroll_event, user_data); #endif SIGNAL_CONNECT(user_data->dlg.draw_area, "key_press_event", key_press_event, user_data); gtk_widget_show(user_data->dlg.draw_area); gtk_widget_show(user_data->dlg.draw_area_comments); gtk_widget_show(viewport); gtk_box_pack_start(GTK_BOX(vbox), user_data->dlg.draw_area, TRUE, TRUE, 0); gtk_widget_show(scroll_window); /* create the associated h_scrollbar */ user_data->dlg.h_scrollbar_adjustment=(GtkAdjustment *)gtk_adjustment_new(0,0,0,0,0,0); user_data->dlg.h_scrollbar=gtk_hscrollbar_new(user_data->dlg.h_scrollbar_adjustment); gtk_widget_show(user_data->dlg.h_scrollbar); gtk_box_pack_end(GTK_BOX(vbox), user_data->dlg.h_scrollbar, FALSE, FALSE, 0); SIGNAL_CONNECT(user_data->dlg.h_scrollbar_adjustment, "value_changed", h_scrollbar_changed, user_data); gtk_box_pack_start(GTK_BOX(hbox), vbox, TRUE, TRUE, 0); gtk_box_pack_start(GTK_BOX(hbox), scroll_window, FALSE, FALSE, 0); /* create the associated v_scrollbar */ user_data->dlg.v_scrollbar_adjustment=(GtkAdjustment *)gtk_adjustment_new(0,0,0,0,0,0); user_data->dlg.v_scrollbar=gtk_vscrollbar_new(user_data->dlg.v_scrollbar_adjustment); gtk_widget_show(user_data->dlg.v_scrollbar); gtk_box_pack_end(GTK_BOX(hbox), user_data->dlg.v_scrollbar, FALSE, FALSE, 0); SIGNAL_CONNECT(user_data->dlg.v_scrollbar_adjustment, "value_changed", v_scrollbar_changed, user_data); gtk_box_pack_start(GTK_BOX(box), hbox, TRUE, TRUE, 0); } /****************************************************************************/ static void dialog_graph_create_window(graph_analysis_data_t* user_data) { GtkWidget *vbox; GtkWidget *hbuttonbox; GtkWidget *bt_close; GtkWidget *bt_save; GtkTooltips *tooltips = gtk_tooltips_new(); /* create the main window */ user_data->dlg.window=window_new(GTK_WINDOW_TOPLEVEL, "Graph Analysis"); vbox=gtk_vbox_new(FALSE, 0); gtk_container_add(GTK_CONTAINER(user_data->dlg.window), vbox); gtk_widget_show(vbox); create_draw_area(user_data, vbox); /* button row */ hbuttonbox = gtk_hbutton_box_new (); gtk_box_pack_start (GTK_BOX (vbox), hbuttonbox, FALSE, FALSE, 0); gtk_button_box_set_layout (GTK_BUTTON_BOX (hbuttonbox), GTK_BUTTONBOX_SPREAD); gtk_button_box_set_spacing (GTK_BUTTON_BOX (hbuttonbox), 30); gtk_widget_show(hbuttonbox); bt_save = BUTTON_NEW_FROM_STOCK(GTK_STOCK_SAVE_AS); gtk_container_add(GTK_CONTAINER(hbuttonbox), bt_save); gtk_widget_show(bt_save); SIGNAL_CONNECT(bt_save, "clicked", on_save_bt_clicked, user_data); gtk_tooltips_set_tip (tooltips, bt_save, "Save an ASCII representation of the graph to a file", NULL); bt_close = BUTTON_NEW_FROM_STOCK(GTK_STOCK_CLOSE); gtk_container_add (GTK_CONTAINER (hbuttonbox), bt_close); GTK_WIDGET_SET_FLAGS(bt_close, GTK_CAN_DEFAULT); gtk_widget_show(bt_close); gtk_tooltips_set_tip (tooltips, bt_close, "Close this dialog", NULL); window_set_cancel_button(user_data->dlg.window, bt_close, window_cancel_button_cb); SIGNAL_CONNECT(user_data->dlg.window, "delete_event", window_delete_event_cb, NULL); SIGNAL_CONNECT(user_data->dlg.window, "destroy", on_destroy, user_data); gtk_widget_show(user_data->dlg.window); window_present(user_data->dlg.window); } /* Return the index array if the node is in the array. Return -1 if there is room in the array * and Return -2 if the array is full */ /****************************************************************************/ gint is_node_array(graph_analysis_data_t* user_data, address* node) { int i; for (i=0; inodes[i].type == AT_NONE) return -1; /* it is not in the array */ if (ADDRESSES_EQUAL((&user_data->nodes[i]),node)) return i; /* it is in the array */ } return -2; /* array full */ } /* Get the nodes from the list */ /****************************************************************************/ void get_nodes(graph_analysis_data_t* user_data) { GList* list; graph_analysis_item_t *gai; gint index; /* fill the node array */ list = g_list_first(user_data->graph_info->list); while (list) { gai = list->data; if (gai->display){ user_data->num_items++; /* check source node address */ index = is_node_array(user_data, &(gai->src_addr)); switch(index){ case -2: /* array full */ gai->src_node = NODE_OVERFLOW; break; case -1: /* not in array */ COPY_ADDRESS(&(user_data->nodes[user_data->num_nodes]),&(gai->src_addr)); gai->src_node = user_data->num_nodes; user_data->num_nodes++; break; default: /* it is in the array, just update the src_node */ gai->src_node = (guint16)index; } /* check destination node address*/ index = is_node_array(user_data, &(gai->dst_addr)); switch(index){ case -2: /* array full */ gai->dst_node = NODE_OVERFLOW; break; case -1: /* not in array */ COPY_ADDRESS(&(user_data->nodes[user_data->num_nodes]),&(gai->dst_addr)); gai->dst_node = user_data->num_nodes; user_data->num_nodes++; break; default: /* it is in the array, just update the dst_node */ gai->dst_node = (guint16)index; } } list = g_list_next(list); } } /****************************************************************************/ graph_analysis_data_t* graph_analysis_init(void) { graph_analysis_data_t* user_data; /* init */ user_data = g_malloc(sizeof(graph_analysis_data_t)); /* init user_data */ graph_analysis_init_dlg(user_data); return user_data; } /****************************************************************************/ void graph_analysis_create(graph_analysis_data_t* user_data) { /* reset the data */ graph_analysis_reset(user_data); /* get nodes (each node is an address) */ get_nodes(user_data); /* create the graph windows */ dialog_graph_create_window(user_data); /* redraw the graph */ dialog_graph_redraw(user_data); return; } /****************************************************************************/ void graph_analysis_update(graph_analysis_data_t* user_data) { /* reset the data */ graph_analysis_reset(user_data); /* get nodes (each node is an address) */ get_nodes(user_data); /* redraw the graph */ dialog_graph_redraw(user_data); window_present(user_data->dlg.window); return; }