2 * Copyright 2004, Irene Ruengeler <i.ruengeler [AT] fh-muenster.de>
3 * Copyright 2009, Varun Notibala <nbvarun [AT] gmail.com>
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.
36 #include "epan/filesystem.h"
37 #include <epan/strutil.h>
39 #include "../globals.h"
41 #include "../ui_util.h"
42 #include "../simple_dialog.h"
44 #include "gtk/dlg_utils.h"
46 #include "gtk/sctp_stat.h"
49 #define DEFAULT_PIXELS_PER_TICK 2
50 #define MAX_PIXELS_PER_TICK 4
51 #define AUTO_MAX_YSCALE 0
52 #define MAX_TICK_VALUES 5
53 #define DEFAULT_TICK_VALUE 3
55 #define MAX_COUNT_TYPES 3
57 #define COUNT_TYPE_FRAMES 0
58 #define COUNT_TYPE_BYTES 1
59 #define COUNT_TYPE_ADVANCED 2
61 #define LEFT_BORDER 60
62 #define RIGHT_BORDER 10
64 #define BOTTOM_BORDER 50
66 #define SUB_32(a, b) a-b
72 * Global variables that help in redrawing graph
75 guint8 gIsSackChunkPresent = 0;
76 guint8 gIsNRSackChunkPresent = 0;
84 struct data_chunk_header {
94 struct init_chunk_header {
110 struct sack_chunk_header {
121 struct nr_sack_chunk_header {
128 guint16 nr_of_nr_gaps;
134 static gboolean label_set = FALSE;
135 static guint32 max_tsn=0, min_tsn=0;
136 static void sctp_graph_set_title(struct sctp_udata *u_data);
137 static void create_draw_area(GtkWidget *box, struct sctp_udata *u_data);
138 static GtkWidget *zoomout_bt;
139 #if defined(_WIN32) && !defined(__MINGW32__)
140 static int rint (double ); /* compiler template for Windows */
143 static void draw_sack_graph(struct sctp_udata *u_data)
146 GList *list=NULL, *tlist;
147 guint16 gap_start=0, gap_end=0, i, j, nr, dup_nr;
149 guint32 tsnumber, dup;
151 GdkColor red_color = {0, 65535, 0, 0};
152 GdkColor green_color = {0, 0, 65535, 0};
153 GdkColor cyan_color = {0, 0, 65535, 65535};
154 GdkGC *red_gc, *green_gc, *cyan_gc;
155 struct sack_chunk_header *sack_header;
157 guint32 max_num, diff;
160 red_gc = gdk_gc_new(u_data->io->draw_area->window);
161 gdk_gc_set_rgb_fg_color(red_gc, &red_color);
163 green_gc = gdk_gc_new(u_data->io->draw_area->window);
164 gdk_gc_set_rgb_fg_color(green_gc, &green_color);
166 cyan_gc = gdk_gc_new(u_data->io->draw_area->window);
167 gdk_gc_set_rgb_fg_color(cyan_gc, &cyan_color);
172 list = g_list_last(u_data->assoc->sack2);
173 if (u_data->io->tmp==FALSE)
175 min_tsn=u_data->assoc->min_tsn2;
176 max_tsn=u_data->assoc->max_tsn2;
180 min_tsn=u_data->assoc->min_tsn2+u_data->io->tmp_min_tsn2;
181 max_tsn=u_data->assoc->min_tsn2+u_data->io->tmp_max_tsn2;
184 else if (u_data->dir==1)
186 list = g_list_last(u_data->assoc->sack1);
187 if (u_data->io->tmp==FALSE)
189 min_tsn=u_data->assoc->min_tsn1;
190 max_tsn=u_data->assoc->max_tsn1;
194 min_tsn=u_data->assoc->min_tsn1+u_data->io->tmp_min_tsn1;
195 max_tsn=u_data->assoc->min_tsn1+u_data->io->tmp_max_tsn1;
201 sack = (tsn_t*) (list->data);
202 tlist = g_list_first(sack->tsns);
205 type = ((struct chunk_header *)tlist->data)->type;
207 if (type == SCTP_SACK_CHUNK_ID)
209 gIsSackChunkPresent = 1;
210 sack_header =(struct sack_chunk_header *)tlist->data;
211 nr=g_ntohs(sack_header->nr_of_gaps);
212 tsnumber = g_ntohl(sack_header->cum_tsn_ack);
213 dup_nr=g_ntohs(sack_header->nr_of_dups);
215 if (sack->secs>=u_data->io->x1_tmp_sec)
219 gap = &sack_header->gaps[0];
222 gap_start=g_ntohs(gap->start);
223 gap_end = g_ntohs(gap->end);
224 max_num=gap_end+tsnumber;
225 for (j=gap_start; j<=gap_end; j++)
227 if (u_data->io->uoff)
228 diff = sack->secs - u_data->io->min_x;
230 diff=sack->secs*1000000+sack->usecs-u_data->io->min_x;
231 xvalue = (guint32)(LEFT_BORDER+u_data->io->offset+u_data->io->x_interval*diff);
232 yvalue = (guint32)(u_data->io->pixmap_height-BOTTOM_BORDER-POINT_SIZE-u_data->io->offset-((SUB_32(j+tsnumber,min_tsn))*u_data->io->y_interval));
233 if (xvalue >= LEFT_BORDER+u_data->io->offset &&
234 xvalue <= u_data->io->pixmap_width-RIGHT_BORDER+u_data->io->offset &&
235 yvalue >= TOP_BORDER-u_data->io->offset-POINT_SIZE &&
236 yvalue <= u_data->io->pixmap_height-BOTTOM_BORDER-u_data->io->offset)
237 gdk_draw_arc(u_data->io->pixmap,green_gc,TRUE,
240 POINT_SIZE, POINT_SIZE,0, (64*360) );
248 if (tsnumber>=min_tsn)
250 if (u_data->io->uoff)
251 diff = sack->secs - u_data->io->min_x;
253 diff=sack->secs*1000000+sack->usecs-u_data->io->min_x;
254 xvalue = (guint32)(LEFT_BORDER+u_data->io->offset+u_data->io->x_interval*diff);
255 yvalue = (guint32)(u_data->io->pixmap_height-BOTTOM_BORDER-POINT_SIZE -u_data->io->offset-((SUB_32(tsnumber,min_tsn))*u_data->io->y_interval));
256 if (xvalue >= LEFT_BORDER+u_data->io->offset &&
257 xvalue <= u_data->io->pixmap_width-RIGHT_BORDER+u_data->io->offset &&
258 yvalue >= TOP_BORDER-u_data->io->offset-POINT_SIZE &&
259 yvalue <= u_data->io->pixmap_height-BOTTOM_BORDER-u_data->io->offset)
261 gdk_draw_arc(u_data->io->pixmap,red_gc,TRUE,
264 POINT_SIZE, POINT_SIZE,0, (64*360) );
269 dup_list = (guint32 *)(((char *)&sack_header->nr_of_dups)+sizeof(guint16)+(nr*sizeof(struct gaps)));
270 for (i = 0; i < dup_nr; i++)
272 dup = g_ntohl(dup_list[i]);
275 if (u_data->io->uoff)
276 diff = sack->secs - u_data->io->min_x;
278 diff=sack->secs*1000000+sack->usecs-u_data->io->min_x;
279 xvalue = (guint32)(LEFT_BORDER+u_data->io->offset+u_data->io->x_interval*diff);
280 yvalue = (guint32)(u_data->io->pixmap_height-BOTTOM_BORDER-POINT_SIZE -u_data->io->offset-((SUB_32(dup,min_tsn))*u_data->io->y_interval));
281 if (xvalue >= LEFT_BORDER+u_data->io->offset &&
282 xvalue <= u_data->io->pixmap_width-RIGHT_BORDER+u_data->io->offset &&
283 yvalue >= TOP_BORDER-u_data->io->offset-POINT_SIZE &&
284 yvalue <= u_data->io->pixmap_height-BOTTOM_BORDER-u_data->io->offset)
285 gdk_draw_arc(u_data->io->pixmap,cyan_gc,TRUE,
288 POINT_SIZE, POINT_SIZE,0, (64*360) );
294 tlist = g_list_next(tlist);
296 list = g_list_previous(list);
298 g_object_unref(G_OBJECT(red_gc));
299 g_object_unref(G_OBJECT(green_gc));
303 * This function plots the NR_SACK gap ack and
305 * Red dot - Cumulative TSN ack
306 * Green dot - Gap ack
307 * Blue circle - NR Gap ack
309 static void draw_nr_sack_graph(struct sctp_udata *u_data)
312 GList *list=NULL, *tlist;
313 guint16 gap_start=0, gap_end=0, i, j, numberOf_gaps, numberOf_nr_gaps;
317 GdkColor red_color = {0, 65535, 0, 0};
318 GdkColor blue_color = {0, 0, 0, 65535};
319 GdkColor green_color = {0, 0, 65535, 0};
320 GdkGC *red_gc, *blue_gc, *green_gc;
321 struct nr_sack_chunk_header *nr_sack_header;
323 guint32 max_num, diff;
324 /* This holds the sum of gap acks and nr gap acks */
325 guint16 total_gaps = 0;
327 red_gc = gdk_gc_new(u_data->io->draw_area->window);
328 gdk_gc_set_rgb_fg_color(red_gc, &red_color);
329 blue_gc = gdk_gc_new(u_data->io->draw_area->window);
330 gdk_gc_set_rgb_fg_color(blue_gc, &blue_color);
331 green_gc = gdk_gc_new(u_data->io->draw_area->window);
332 gdk_gc_set_rgb_fg_color(green_gc, &green_color);
337 list = g_list_last(u_data->assoc->sack2);
338 if (u_data->io->tmp==FALSE)
340 min_tsn=u_data->assoc->min_tsn2;
341 max_tsn=u_data->assoc->max_tsn2;
345 min_tsn=u_data->assoc->min_tsn2+u_data->io->tmp_min_tsn2;
346 max_tsn=u_data->assoc->min_tsn2+u_data->io->tmp_max_tsn2;
349 else if (u_data->dir==1)
351 list = g_list_last(u_data->assoc->sack1);
352 if (u_data->io->tmp==FALSE)
354 min_tsn=u_data->assoc->min_tsn1;
355 max_tsn=u_data->assoc->max_tsn1;
359 min_tsn=u_data->assoc->min_tsn1+u_data->io->tmp_min_tsn1;
360 max_tsn=u_data->assoc->min_tsn1+u_data->io->tmp_max_tsn1;
365 sack = (tsn_t*) (list->data);
366 tlist = g_list_first(sack->tsns);
369 type = ((struct chunk_header *)tlist->data)->type;
371 * The tlist->data is memcpy ied to the appropriate structure
372 * They entire raw tvb bytes are copied on to one of the *_chunk_header
373 * structures in sctp_stat.c
375 if (type == SCTP_NR_SACK_CHUNK_ID)
377 gIsNRSackChunkPresent = 1;
378 nr_sack_header =(struct nr_sack_chunk_header *)tlist->data;
379 numberOf_nr_gaps=g_ntohs(nr_sack_header->nr_of_nr_gaps);
380 numberOf_gaps=g_ntohs(nr_sack_header->nr_of_gaps);
381 tsnumber = g_ntohl(nr_sack_header->cum_tsn_ack);
382 total_gaps = numberOf_gaps + numberOf_nr_gaps;
383 if (sack->secs>=u_data->io->x1_tmp_sec)
385 /* If the number of nr_gaps is greater than 0 */
386 if ( total_gaps > 0 )
388 nr_gap = &nr_sack_header->gaps[0];
389 for ( i=0; i < total_gaps; i++ )
391 gap_start=g_ntohs(nr_gap->start);
392 gap_end = g_ntohs(nr_gap->end);
393 max_num= gap_end + tsnumber;
394 for ( j = gap_start; j <= gap_end; j++)
396 if (u_data->io->uoff)
397 diff = sack->secs - u_data->io->min_x;
399 diff=sack->secs*1000000+sack->usecs-u_data->io->min_x;
400 xvalue = (guint32)(LEFT_BORDER+u_data->io->offset+u_data->io->x_interval*diff);
401 yvalue = (guint32)(u_data->io->pixmap_height-BOTTOM_BORDER-POINT_SIZE-u_data->io->offset-((SUB_32(j+tsnumber,min_tsn))*u_data->io->y_interval));
402 if (xvalue >= LEFT_BORDER+u_data->io->offset &&
403 xvalue <= u_data->io->pixmap_width-RIGHT_BORDER+u_data->io->offset &&
404 yvalue >= TOP_BORDER-u_data->io->offset-POINT_SIZE &&
405 yvalue <= u_data->io->pixmap_height-BOTTOM_BORDER-u_data->io->offset)
407 /* Check if this is an GAP ACK or NR GAP ACK */
408 if ( i >= numberOf_gaps)
410 /* This is a nr gap ack */
411 gdk_draw_arc(u_data->io->pixmap,blue_gc,FALSE,
414 POINT_SIZE , POINT_SIZE,0, (64*360) );
416 /* All NR GAP Acks are also gap acks, so plot these as
417 * gap acks - green dot.
418 * These will be shown as points with a green dot - GAP ACK
419 * surrounded by a blue circle - NR GAP ack
421 gdk_draw_arc(u_data->io->pixmap, green_gc, TRUE,
424 POINT_SIZE , POINT_SIZE,0, (64*360) );
429 /* This is just a gap ack */
430 gdk_draw_arc(u_data->io->pixmap, green_gc, TRUE,
433 POINT_SIZE , POINT_SIZE,0, (64*360) );
437 if (i < total_gaps-1)
444 if (tsnumber>=min_tsn)
446 if (u_data->io->uoff)
447 diff = sack->secs - u_data->io->min_x;
449 diff=sack->secs*1000000+sack->usecs-u_data->io->min_x;
450 xvalue = (guint32)(LEFT_BORDER+u_data->io->offset+u_data->io->x_interval*diff);
451 yvalue = (guint32)(u_data->io->pixmap_height-BOTTOM_BORDER-POINT_SIZE -u_data->io->offset-((SUB_32(tsnumber,min_tsn))*u_data->io->y_interval));
452 if (xvalue >= LEFT_BORDER+u_data->io->offset &&
453 xvalue <= u_data->io->pixmap_width-RIGHT_BORDER+u_data->io->offset &&
454 yvalue >= TOP_BORDER-u_data->io->offset-POINT_SIZE &&
455 yvalue <= u_data->io->pixmap_height-BOTTOM_BORDER-u_data->io->offset)
456 gdk_draw_arc(u_data->io->pixmap,red_gc,TRUE,
459 POINT_SIZE, POINT_SIZE,0, (64*360) );
463 tlist = g_list_next(tlist);
465 list = g_list_previous(list);
467 g_object_unref(G_OBJECT(red_gc));
468 g_object_unref(G_OBJECT(blue_gc));
469 g_object_unref(G_OBJECT(green_gc));
472 static void draw_tsn_graph(struct sctp_udata *u_data)
475 GList *list=NULL, *tlist;
478 guint32 min_secs=0, diff;
483 list = g_list_last(u_data->assoc->tsn1);
484 if (u_data->io->tmp==FALSE)
486 min_tsn=u_data->assoc->min_tsn1;
487 max_tsn=u_data->assoc->max_tsn1;
491 min_tsn=u_data->assoc->min_tsn1+u_data->io->tmp_min_tsn1;
492 max_tsn=u_data->assoc->min_tsn1+u_data->io->tmp_max_tsn1;
495 else if (u_data->dir==2)
497 list = g_list_last(u_data->assoc->tsn2);
498 if (u_data->io->tmp==FALSE)
500 min_tsn=u_data->assoc->min_tsn2;
501 max_tsn=u_data->assoc->max_tsn2;
505 min_tsn=u_data->assoc->min_tsn2+u_data->io->tmp_min_tsn2;
506 max_tsn=u_data->assoc->min_tsn2+u_data->io->tmp_max_tsn2;
512 tsn = (tsn_t*) (list->data);
513 tlist = g_list_first(tsn->tsns);
516 type = ((struct chunk_header *)tlist->data)->type;
517 if (type == SCTP_DATA_CHUNK_ID)
518 tsnumber = g_ntohl(((struct data_chunk_header *)tlist->data)->tsn);
519 if (tsnumber>=min_tsn && tsnumber<=max_tsn && tsn->secs>=min_secs)
521 if (u_data->io->uoff)
522 diff = tsn->secs - u_data->io->min_x;
524 diff=tsn->secs*1000000+tsn->usecs-u_data->io->min_x;
525 xvalue = (guint32)(LEFT_BORDER+u_data->io->offset+u_data->io->x_interval*diff);
526 yvalue = (guint32)(u_data->io->pixmap_height-BOTTOM_BORDER-POINT_SIZE-u_data->io->offset-((SUB_32(tsnumber,min_tsn))*u_data->io->y_interval));
527 if (xvalue >= LEFT_BORDER+u_data->io->offset &&
528 xvalue <= u_data->io->pixmap_width-RIGHT_BORDER+u_data->io->offset &&
529 yvalue >= TOP_BORDER-u_data->io->offset-POINT_SIZE &&
530 yvalue <= u_data->io->pixmap_height-BOTTOM_BORDER-u_data->io->offset)
531 gdk_draw_arc(u_data->io->pixmap,u_data->io->draw_area->style->black_gc,TRUE,
534 POINT_SIZE, POINT_SIZE, 0, (64*360));
536 tlist = g_list_next(tlist);
538 list = g_list_previous(list);
543 static void sctp_graph_draw(struct sctp_udata *u_data)
546 guint32 distance=5, i, e, sec, w, start, a, b, j;
547 gint label_width, label_height;
548 char label_string[15];
550 gboolean write_label = FALSE;
553 if (u_data->io->x1_tmp_sec==0 && u_data->io->x1_tmp_usec==0)
554 u_data->io->offset=0;
556 u_data->io->offset=5;
558 if (u_data->io->x2_tmp_sec - u_data->io->x1_tmp_sec > 1500)
560 u_data->io->min_x=u_data->io->x1_tmp_sec;
561 u_data->io->max_x=u_data->io->x2_tmp_sec;
562 u_data->io->uoff = TRUE;
566 u_data->io->min_x=(guint32)(u_data->io->x1_tmp_sec*1000000.0+u_data->io->x1_tmp_usec);
567 u_data->io->max_x=(guint32)(u_data->io->x2_tmp_sec*1000000.0+u_data->io->x2_tmp_usec);
568 u_data->io->uoff = FALSE;
571 u_data->io->tmp_width=u_data->io->max_x-u_data->io->min_x;
575 if (u_data->io->tmp==FALSE)
577 if (u_data->assoc->tsn1!=NULL || u_data->assoc->sack1!=NULL)
578 u_data->io->max_y=u_data->io->tmp_max_tsn1 - u_data->io->tmp_min_tsn1;
580 u_data->io->max_y= 0;
581 u_data->io->min_y = 0;
585 u_data->io->max_y = u_data->io->tmp_max_tsn1;
586 u_data->io->min_y = u_data->io->tmp_min_tsn1;
589 else if (u_data->dir==2)
591 if (u_data->io->tmp==FALSE)
593 if (u_data->assoc->tsn2!=NULL || u_data->assoc->sack2!=NULL)
594 u_data->io->max_y=u_data->io->tmp_max_tsn2 -u_data->io->tmp_min_tsn2;
596 u_data->io->max_y= 0;
597 u_data->io->min_y = 0;
601 u_data->io->max_y = u_data->io->tmp_max_tsn2;
602 u_data->io->min_y = u_data->io->tmp_min_tsn2;
606 gdk_draw_rectangle(u_data->io->pixmap,
607 u_data->io->draw_area->style->white_gc,
610 u_data->io->draw_area->allocation.width,
611 u_data->io->draw_area->allocation.height);
615 gdk_draw_line(u_data->io->pixmap,u_data->io->draw_area->style->black_gc, LEFT_BORDER+u_data->io->offset,u_data->io->pixmap_height-BOTTOM_BORDER,u_data->io->pixmap_width-RIGHT_BORDER+u_data->io->offset, u_data->io->pixmap_height-BOTTOM_BORDER);
616 gdk_draw_line(u_data->io->pixmap,u_data->io->draw_area->style->black_gc,u_data->io->pixmap_width-RIGHT_BORDER+u_data->io->offset, u_data->io->pixmap_height-BOTTOM_BORDER, u_data->io->pixmap_width-RIGHT_BORDER+u_data->io->offset-5, u_data->io->pixmap_height-BOTTOM_BORDER-5);
617 gdk_draw_line(u_data->io->pixmap,u_data->io->draw_area->style->black_gc,u_data->io->pixmap_width-RIGHT_BORDER+u_data->io->offset, u_data->io->pixmap_height-BOTTOM_BORDER, u_data->io->pixmap_width-RIGHT_BORDER+u_data->io->offset-5, u_data->io->pixmap_height-BOTTOM_BORDER+5);
618 u_data->io->axis_width=u_data->io->pixmap_width-LEFT_BORDER-RIGHT_BORDER-u_data->io->offset;
620 /* try to avoid dividing by zero */
621 if(u_data->io->tmp_width>0){
622 u_data->io->x_interval = (float)((u_data->io->axis_width*1.0)/u_data->io->tmp_width); /*distance in pixels between 2 data points*/
624 u_data->io->x_interval = (float)(u_data->io->axis_width);
627 e=0; /*number of decimals of x_interval*/
628 if (u_data->io->x_interval<1)
630 dis=1/u_data->io->x_interval;
637 for (i=0; i<=e+1; i++)
638 distance*=10; /*distance per 100 pixels*/
643 g_snprintf(label_string, sizeof(label_string), "%d", 0);
644 memcpy(label_string,(gchar *)g_locale_to_utf8(label_string, -1 , NULL, NULL, NULL), sizeof(label_string));
645 layout = gtk_widget_create_pango_layout(u_data->io->draw_area, label_string);
646 pango_layout_get_pixel_size(layout, &label_width, &label_height);
648 if (u_data->io->x1_tmp_usec==0)
649 sec=u_data->io->x1_tmp_sec;
651 sec=u_data->io->x1_tmp_sec+1;
654 if (u_data->io->offset!=0)
656 g_snprintf(label_string, sizeof(label_string), "%u", u_data->io->x1_tmp_sec);
658 memcpy(label_string,(gchar *)g_locale_to_utf8(label_string, -1 , NULL, NULL, NULL), sizeof(label_string));
659 pango_layout_set_text(layout, label_string, -1);
660 pango_layout_get_pixel_size(layout, &lwidth, NULL);
662 gdk_draw_layout(u_data->io->pixmap,u_data->io->draw_area->style->black_gc,
664 u_data->io->pixmap_height-BOTTOM_BORDER+20,
668 w=(guint32)(500/(guint32)(distance*u_data->io->x_interval)); /*there will be a label for every w_th tic*/
673 if (w==4 || w==3 || w==2)
676 a=distance/10; /*distance between two tics*/
677 b = (guint32)((u_data->io->min_x/100000))%10; /* start for labels*/
686 if (!u_data->io->uoff)
690 start=u_data->io->min_x/1000000*1000000;
696 start=u_data->io->min_x/100000;
700 b = (guint32)((start/100000))%10;
705 start = u_data->io->min_x;
712 for (i=start, j=b; i<=u_data->io->max_x; i+=a, j++)
714 if (!u_data->io->uoff)
715 if (i>=u_data->io->min_x && i%1000000!=0)
718 g_snprintf(label_string, sizeof(label_string), "%d", i%1000000);
722 memcpy(label_string,(gchar *)g_locale_to_utf8(label_string, -1 , NULL, NULL, NULL), sizeof(label_string));
723 pango_layout_set_text(layout, label_string, -1);
724 pango_layout_get_pixel_size(layout, &lwidth, NULL);
725 gdk_draw_layout(u_data->io->pixmap,u_data->io->draw_area->style->black_gc,
726 (guint32)(LEFT_BORDER+u_data->io->offset+(i-u_data->io->min_x)*u_data->io->x_interval-lwidth/2),
727 u_data->io->pixmap_height-BOTTOM_BORDER+10,
730 gdk_draw_line(u_data->io->pixmap,u_data->io->draw_area->style->black_gc,
731 (guint32)(LEFT_BORDER+u_data->io->offset+(i-u_data->io->min_x)*u_data->io->x_interval),
732 u_data->io->pixmap_height-BOTTOM_BORDER,
733 (guint32)(LEFT_BORDER+u_data->io->offset+(i-u_data->io->min_x)*u_data->io->x_interval),
734 u_data->io->pixmap_height-BOTTOM_BORDER+length);
737 if (!u_data->io->uoff)
739 if (i%1000000==0 && j%w==0)
755 gdk_draw_line(u_data->io->pixmap,u_data->io->draw_area->style->black_gc,
756 (guint32)(LEFT_BORDER+u_data->io->offset+(i-u_data->io->min_x)*u_data->io->x_interval),
757 u_data->io->pixmap_height-BOTTOM_BORDER,
758 (guint32)(LEFT_BORDER+u_data->io->offset+(i-u_data->io->min_x)*u_data->io->x_interval),
759 u_data->io->pixmap_height-BOTTOM_BORDER+10);
761 g_snprintf(label_string, sizeof(label_string), "%d", sec);
762 memcpy(label_string,(gchar *)g_locale_to_utf8(label_string, -1 , NULL, NULL, NULL), sizeof(label_string));
763 pango_layout_set_text(layout, label_string, -1);
764 pango_layout_get_pixel_size(layout, &lwidth, NULL);
766 gdk_draw_layout(u_data->io->pixmap,u_data->io->draw_area->style->black_gc,
767 (guint32)(LEFT_BORDER+u_data->io->offset+(i-u_data->io->min_x)*u_data->io->x_interval-10),
768 u_data->io->pixmap_height-BOTTOM_BORDER+20,
775 g_strlcpy(label_string, "sec", sizeof(label_string));
777 memcpy(label_string,(gchar *)g_locale_to_utf8(label_string, -1 , NULL, NULL, NULL), sizeof(label_string));
778 pango_layout_set_text(layout, label_string, -1);
779 pango_layout_get_pixel_size(layout, &lwidth, NULL);
780 gdk_draw_layout(u_data->io->pixmap,
781 u_data->io->draw_area->style->black_gc,
782 u_data->io->pixmap_width-RIGHT_BORDER-10,
783 u_data->io->pixmap_height-BOTTOM_BORDER+30,
789 gdk_draw_line(u_data->io->pixmap,u_data->io->draw_area->style->black_gc, LEFT_BORDER,TOP_BORDER-u_data->io->offset,LEFT_BORDER,u_data->io->pixmap_height-BOTTOM_BORDER-u_data->io->offset);
790 gdk_draw_line(u_data->io->pixmap,u_data->io->draw_area->style->black_gc,LEFT_BORDER,TOP_BORDER-u_data->io->offset, LEFT_BORDER-5, TOP_BORDER-u_data->io->offset+5);
791 gdk_draw_line(u_data->io->pixmap,u_data->io->draw_area->style->black_gc,LEFT_BORDER,TOP_BORDER-u_data->io->offset, LEFT_BORDER+5, TOP_BORDER-u_data->io->offset+5);
793 u_data->io->y_interval = (float)(((u_data->io->pixmap_height-TOP_BORDER-BOTTOM_BORDER)*1.0)/(u_data->io->max_y-u_data->io->min_y));
796 if (u_data->io->y_interval<1)
798 dis=1/u_data->io->y_interval;
806 distance=distance*10;
808 else if (u_data->io->y_interval<2)
811 if (u_data->io->max_y>0)
813 for (i=u_data->io->min_y/distance*distance; i<=u_data->io->max_y; i+=distance/5)
815 if (i>=u_data->io->min_y)
818 g_snprintf(label_string, sizeof(label_string), "%d", i);
819 if (i%distance==0 || (distance<=5 && u_data->io->y_interval>10))
823 memcpy(label_string,(gchar *)g_locale_to_utf8(label_string, -1 , NULL, NULL, NULL), sizeof(label_string));
824 pango_layout_set_text(layout, label_string, -1);
825 pango_layout_get_pixel_size(layout, &lwidth, NULL);
826 gdk_draw_layout(u_data->io->pixmap,u_data->io->draw_area->style->black_gc,
827 LEFT_BORDER-length-lwidth-5,
828 (guint32)(u_data->io->pixmap_height-BOTTOM_BORDER-u_data->io->offset-(i-u_data->io->min_y)*u_data->io->y_interval-POINT_SIZE),
831 gdk_draw_line(u_data->io->pixmap,u_data->io->draw_area->style->black_gc,LEFT_BORDER-length,
832 (guint32)(u_data->io->pixmap_height-BOTTOM_BORDER-u_data->io->offset-(i-u_data->io->min_y)*u_data->io->y_interval),
834 (guint32)(u_data->io->pixmap_height-BOTTOM_BORDER-u_data->io->offset-(i-u_data->io->min_y)*u_data->io->y_interval));
838 else if ((u_data->dir==1 && u_data->assoc->n_array_tsn1==0) || (u_data->dir==2 && u_data->assoc->n_array_tsn2==0))
839 simple_dialog(ESD_TYPE_INFO, ESD_BTN_OK, "No Data Chunks sent");
842 /* This function is used to change the title
843 * in the graph dialogue to NR_SACK or SACK based on the
845 * If an association has both SAKC and NR_SACK PDU's
846 * a warning is popped
848 void updateLabels(void)
850 if (gIsSackChunkPresent && gIsNRSackChunkPresent)
852 simple_dialog(ESD_TYPE_INFO, ESD_BTN_OK, "This data set contains both SACK and NR SACK PDUs.");
853 gtk_button_set_label( (GtkButton*) sack_bt, "Show both Sack and NR Sack");
855 else if (gIsSackChunkPresent)
856 gtk_button_set_label( (GtkButton*) sack_bt, "Show Only Sack");
858 /* gIsNRSackChunkPresent will be true here */
859 gtk_button_set_label( (GtkButton*) sack_bt, "Show Only NR Sack");
862 static void sctp_graph_redraw(struct sctp_udata *u_data)
866 u_data->io->needs_redraw=TRUE;
868 sctp_graph_draw(u_data);
869 switch (u_data->io->graph_type)
872 /* Show both TSN and SACK information
873 * Reset the global sack variable
874 * for sack and nr sack cases
876 gIsSackChunkPresent = 0;
877 gIsNRSackChunkPresent = 0;
878 draw_sack_graph(u_data);
879 draw_nr_sack_graph(u_data);
880 draw_tsn_graph(u_data);
884 draw_tsn_graph(u_data);
887 /* Show only SACK information
888 * Reset the global sack variable
889 * for sack and nr sack cases
891 gIsSackChunkPresent = 0;
892 gIsNRSackChunkPresent = 0;
893 draw_sack_graph(u_data);
894 draw_nr_sack_graph(u_data);
898 /* Updates the sack / nr sack buttons */
901 ios=(sctp_graph_t *)g_object_get_data(G_OBJECT(u_data->io->draw_area), "sctp_graph_t");
907 gdk_draw_pixmap(u_data->io->draw_area->window,
908 u_data->io->draw_area->style->fg_gc[GTK_WIDGET_STATE(u_data->io->draw_area)],
912 u_data->io->draw_area->allocation.width,
913 u_data->io->draw_area->allocation.height);
917 static void on_sack_bt(GtkWidget *widget _U_, struct sctp_udata *u_data)
920 u_data = (struct sctp_udata *) u_data;
921 u_data->io->graph_type=2;
922 sctp_graph_redraw(u_data);
925 static void on_tsn_bt(GtkWidget *widget _U_, struct sctp_udata *u_data)
928 u_data->io->graph_type=1;
929 sctp_graph_redraw(u_data);
932 static void on_both_bt(GtkWidget *widget _U_, struct sctp_udata *u_data)
935 u_data->io->graph_type=0;
936 sctp_graph_redraw(u_data);
940 sctp_graph_close_cb(GtkWidget* widget _U_, gpointer u_data)
942 struct sctp_udata *udata;
945 udata = (struct sctp_udata *)u_data;
947 gtk_grab_remove(GTK_WIDGET(udata->io->window));
948 gtk_widget_destroy(GTK_WIDGET(udata->io->window));
953 configure_event(GtkWidget *widget, GdkEventConfigure *event _U_, struct sctp_udata *u_data)
959 if(u_data->io->pixmap){
960 gdk_pixmap_unref(u_data->io->pixmap);
961 u_data->io->pixmap=NULL;
964 u_data->io->pixmap=gdk_pixmap_new(widget->window,
965 widget->allocation.width,
966 widget->allocation.height,
968 u_data->io->pixmap_width=widget->allocation.width;
969 u_data->io->pixmap_height=widget->allocation.height;
971 gdk_draw_rectangle(u_data->io->pixmap,
972 widget->style->white_gc,
975 widget->allocation.width,
976 widget->allocation.height);
977 sctp_graph_redraw(u_data);
982 expose_event(GtkWidget *widget, GdkEventExpose *event)
986 ios=(sctp_graph_t *)g_object_get_data(G_OBJECT(widget), "sctp_graph_t");
991 gdk_draw_pixmap(widget->window,
992 widget->style->fg_gc[GTK_WIDGET_STATE(widget)],
994 event->area.x, event->area.y,
995 event->area.x, event->area.y,
996 event->area.width, event->area.height);
1003 on_zoomin_bt (GtkWidget *widget _U_, struct sctp_udata *u_data)
1005 sctp_min_max_t *tmp_minmax;
1007 if (u_data->io->rectangle_present==TRUE)
1009 tmp_minmax = g_malloc(sizeof(sctp_min_max_t));
1011 u_data->io->tmp_min_tsn1=u_data->io->y1_tmp+u_data->io->min_y;
1012 u_data->io->tmp_max_tsn1=u_data->io->y2_tmp+1+u_data->io->min_y;
1013 u_data->io->tmp_min_tsn2=u_data->io->tmp_min_tsn1;
1014 u_data->io->tmp_max_tsn2=u_data->io->tmp_max_tsn1;
1015 tmp_minmax->tmp_min_secs=u_data->io->x1_tmp_sec;
1016 tmp_minmax->tmp_min_usecs= u_data->io->x1_tmp_usec;
1017 tmp_minmax->tmp_max_secs= u_data->io->x2_tmp_sec;
1018 tmp_minmax->tmp_max_usecs= u_data->io->x2_tmp_usec;
1019 tmp_minmax->tmp_min_tsn1=u_data->io->tmp_min_tsn1;
1020 tmp_minmax->tmp_max_tsn1=u_data->io->tmp_max_tsn1;
1021 tmp_minmax->tmp_min_tsn2=u_data->io->tmp_min_tsn2;
1022 tmp_minmax->tmp_max_tsn2=u_data->io->tmp_max_tsn2;
1023 u_data->assoc->min_max = g_slist_prepend(u_data->assoc->min_max, tmp_minmax);
1024 u_data->io->length = g_slist_length(u_data->assoc->min_max);
1025 u_data->io->tmp=TRUE;
1026 u_data->io->rectangle=FALSE;
1027 u_data->io->rectangle_present=FALSE;
1028 gtk_widget_set_sensitive(zoomout_bt, TRUE);
1029 sctp_graph_redraw(u_data);
1033 simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, "Please draw a rectangle around the area you want to zoom in!");
1038 zoomin_bt_fcn (struct sctp_udata *u_data)
1040 sctp_min_max_t *tmp_minmax;
1042 tmp_minmax = g_malloc(sizeof(sctp_min_max_t));
1044 u_data->io->tmp_min_tsn1=u_data->io->y1_tmp+u_data->io->min_y;
1045 u_data->io->tmp_max_tsn1=u_data->io->y2_tmp+1+u_data->io->min_y;
1046 u_data->io->tmp_min_tsn2=u_data->io->tmp_min_tsn1;
1047 u_data->io->tmp_max_tsn2=u_data->io->tmp_max_tsn1;
1048 tmp_minmax->tmp_min_secs=u_data->io->x1_tmp_sec;
1049 tmp_minmax->tmp_min_usecs= u_data->io->x1_tmp_usec;
1050 tmp_minmax->tmp_max_secs= u_data->io->x2_tmp_sec;
1051 tmp_minmax->tmp_max_usecs= u_data->io->x2_tmp_usec;
1052 tmp_minmax->tmp_min_tsn1=u_data->io->tmp_min_tsn1;
1053 tmp_minmax->tmp_max_tsn1=u_data->io->tmp_max_tsn1;
1054 tmp_minmax->tmp_min_tsn2=u_data->io->tmp_min_tsn2;
1055 tmp_minmax->tmp_max_tsn2=u_data->io->tmp_max_tsn2;
1056 u_data->assoc->min_max = g_slist_prepend(u_data->assoc->min_max, tmp_minmax);
1057 u_data->io->length = g_slist_length(u_data->assoc->min_max);
1058 u_data->io->tmp=TRUE;
1059 u_data->io->rectangle=FALSE;
1060 u_data->io->rectangle_present=FALSE;
1061 gtk_widget_set_sensitive(zoomout_bt, TRUE);
1062 sctp_graph_redraw(u_data);
1069 on_zoomout_bt (GtkWidget *widget _U_, struct sctp_udata *u_data)
1071 sctp_min_max_t *tmp_minmax, *mm;
1074 l = g_slist_length(u_data->assoc->min_max);
1076 if (u_data->assoc->min_max!=NULL)
1078 mm=(sctp_min_max_t *)((u_data->assoc->min_max)->data);
1079 u_data->assoc->min_max=g_slist_remove(u_data->assoc->min_max, mm);
1083 tmp_minmax = (sctp_min_max_t *)u_data->assoc->min_max->data;
1084 u_data->io->x1_tmp_sec=tmp_minmax->tmp_min_secs;
1085 u_data->io->x1_tmp_usec=tmp_minmax->tmp_min_usecs;
1086 u_data->io->x2_tmp_sec=tmp_minmax->tmp_max_secs;
1087 u_data->io->x2_tmp_usec=tmp_minmax->tmp_max_usecs;
1088 u_data->io->tmp_min_tsn1=tmp_minmax->tmp_min_tsn1;
1089 u_data->io->tmp_max_tsn1=tmp_minmax->tmp_max_tsn1;
1090 u_data->io->tmp_min_tsn2=tmp_minmax->tmp_min_tsn2;
1091 u_data->io->tmp_max_tsn2=tmp_minmax->tmp_max_tsn2;
1092 u_data->io->tmp=TRUE;
1096 u_data->io->x1_tmp_sec=u_data->assoc->min_secs;
1097 u_data->io->x1_tmp_usec=u_data->assoc->min_usecs;
1098 u_data->io->x2_tmp_sec=u_data->assoc->max_secs;
1099 u_data->io->x2_tmp_usec=u_data->assoc->max_usecs;
1100 u_data->io->tmp_min_tsn1=u_data->assoc->min_tsn1;
1101 u_data->io->tmp_max_tsn1=u_data->assoc->max_tsn1;
1102 u_data->io->tmp_min_tsn2=u_data->assoc->min_tsn2;
1103 u_data->io->tmp_max_tsn2=u_data->assoc->max_tsn2;
1104 u_data->io->tmp=FALSE;
1109 u_data->io->x1_tmp_sec=u_data->assoc->min_secs;
1110 u_data->io->x1_tmp_usec=u_data->assoc->min_usecs;
1111 u_data->io->x2_tmp_sec=u_data->assoc->max_secs;
1112 u_data->io->x2_tmp_usec=u_data->assoc->max_usecs;
1113 u_data->io->tmp_min_tsn1=u_data->assoc->min_tsn1;
1114 u_data->io->tmp_max_tsn1=u_data->assoc->max_tsn1;
1115 u_data->io->tmp_min_tsn2=u_data->assoc->min_tsn2;
1116 u_data->io->tmp_max_tsn2=u_data->assoc->max_tsn2;
1117 u_data->io->tmp=FALSE;
1119 if (g_slist_length(u_data->assoc->min_max)==1)
1120 gtk_widget_set_sensitive(zoomout_bt, FALSE);
1121 sctp_graph_redraw(u_data);
1125 on_button_press (GtkWidget *widget _U_, GdkEventButton *event, struct sctp_udata *u_data)
1129 if (u_data->io->rectangle==TRUE)
1131 gdk_draw_rectangle(u_data->io->pixmap,u_data->io->draw_area->style->white_gc,
1133 (gint)floor(MIN(u_data->io->x_old,u_data->io->x_new)),
1134 (gint)floor(MIN(u_data->io->y_old,u_data->io->y_new)),
1135 (gint)floor(abs((long)(u_data->io->x_new-u_data->io->x_old))),
1136 (gint)floor(abs((long)(u_data->io->y_new-u_data->io->y_old))));
1137 ios=(sctp_graph_t *)g_object_get_data(G_OBJECT(u_data->io->draw_area), "sctp_graph_t");
1143 gdk_draw_pixmap(u_data->io->draw_area->window,
1144 u_data->io->draw_area->style->fg_gc[GTK_WIDGET_STATE(u_data->io->draw_area)],
1148 (gint)(abs((long)(u_data->io->x_new-u_data->io->x_old))),
1149 (gint)(abs((long)(u_data->io->y_new-u_data->io->y_old))));
1150 sctp_graph_redraw(u_data);
1152 u_data->io->x_old=event->x;
1153 u_data->io->y_old=event->y;
1154 if (u_data->io->y_old>u_data->io->pixmap_height-BOTTOM_BORDER-u_data->io->offset-POINT_SIZE)
1155 u_data->io->y_old=u_data->io->pixmap_height-BOTTOM_BORDER-u_data->io->offset-POINT_SIZE;
1156 if (u_data->io->x_old<LEFT_BORDER+u_data->io->offset)
1157 u_data->io->x_old=LEFT_BORDER+u_data->io->offset;
1158 u_data->io->rectangle=FALSE;
1165 on_button_release (GtkWidget *widget _U_, GdkEventButton *event, struct sctp_udata *u_data)
1168 guint32 helpx, helpy, x1_tmp, x2_tmp, y_value, t_size=0, s_size=0, i, y_tolerance;
1169 gint label_width, label_height;
1170 gdouble x_value, position, s_diff=0, t_diff=0, x_tolerance=0.0001;
1172 char label_string[30];
1174 GPtrArray *tsnlist = NULL, *sacklist=NULL;
1175 struct tsn_sort *tsn, *sack=NULL;
1176 gboolean sack_found = FALSE;
1177 PangoLayout *layout;
1179 g_snprintf(label_string, 15, "%d", 0);
1180 memcpy(label_string,(gchar *)g_locale_to_utf8(label_string, -1 , NULL, NULL, NULL), 15);
1181 layout = gtk_widget_create_pango_layout(u_data->io->draw_area, label_string);
1182 pango_layout_get_pixel_size(layout, &label_width, &label_height);
1184 if (event->y>u_data->io->pixmap_height-BOTTOM_BORDER-u_data->io->offset)
1185 event->y = u_data->io->pixmap_height-BOTTOM_BORDER-u_data->io->offset;
1186 if (event->x < LEFT_BORDER+u_data->io->offset)
1187 event->x = LEFT_BORDER+u_data->io->offset;
1188 if (abs((long)(event->x-u_data->io->x_old))>10 || abs((long)(event->y-u_data->io->y_old))>10)
1190 u_data->io->rect_x_min = (gint)floor(MIN(u_data->io->x_old,event->x));
1191 u_data->io->rect_x_max = (gint)ceil(MAX(u_data->io->x_old,event->x));
1192 u_data->io->rect_y_min = (gint)floor(MIN(u_data->io->y_old,event->y));
1193 u_data->io->rect_y_max = (gint)ceil(MAX(u_data->io->y_old,event->y))+POINT_SIZE;
1194 gdk_draw_rectangle(u_data->io->pixmap,u_data->io->draw_area->style->black_gc,
1196 u_data->io->rect_x_min, u_data->io->rect_y_min,
1197 u_data->io->rect_x_max - u_data->io->rect_x_min,
1198 u_data->io->rect_y_max - u_data->io->rect_y_min);
1199 ios=(sctp_graph_t *)g_object_get_data(G_OBJECT(u_data->io->draw_area), "sctp_graph_t");
1205 gdk_draw_pixmap(u_data->io->draw_area->window,
1206 u_data->io->draw_area->style->fg_gc[GTK_WIDGET_STATE(u_data->io->draw_area)],
1210 u_data->io->draw_area->allocation.width,
1211 u_data->io->draw_area->allocation.height);
1213 x1_tmp=(unsigned int)floor(u_data->io->min_x+((u_data->io->x_old-LEFT_BORDER-u_data->io->offset)*u_data->io->tmp_width/u_data->io->axis_width));
1214 x2_tmp=(unsigned int)floor(u_data->io->min_x+((event->x-LEFT_BORDER-u_data->io->offset)*u_data->io->tmp_width/u_data->io->axis_width));
1215 helpx=MIN(x1_tmp, x2_tmp);
1221 if (u_data->io->uoff)
1223 if (x2_tmp - x1_tmp <= 1500)
1224 u_data->io->uoff = FALSE;
1225 u_data->io->x1_tmp_sec=(guint32)x1_tmp;
1226 u_data->io->x1_tmp_usec=0;
1227 u_data->io->x2_tmp_sec=(guint32)x2_tmp;
1228 u_data->io->x2_tmp_usec=0;
1232 u_data->io->x1_tmp_sec=(guint32)x1_tmp/1000000;
1233 u_data->io->x1_tmp_usec=x1_tmp%1000000;
1234 u_data->io->x2_tmp_sec=(guint32)x2_tmp/1000000;
1235 u_data->io->x2_tmp_usec=x2_tmp%1000000;
1237 u_data->io->x1_akt_sec = u_data->io->x1_tmp_sec;
1238 u_data->io->x1_akt_usec = u_data->io->x1_tmp_usec;
1239 u_data->io->x2_akt_sec = u_data->io->x2_tmp_sec;
1240 u_data->io->x2_akt_usec = u_data->io->x2_tmp_usec;
1242 u_data->io->y1_tmp=(guint32)((u_data->io->pixmap_height-BOTTOM_BORDER-u_data->io->offset-u_data->io->y_old)/u_data->io->y_interval);
1243 u_data->io->y2_tmp=(guint32)((u_data->io->pixmap_height-BOTTOM_BORDER-u_data->io->offset-event->y)/u_data->io->y_interval);
1244 helpy = MIN(u_data->io->y1_tmp, u_data->io->y2_tmp);
1245 u_data->io->y2_tmp = MAX(u_data->io->y1_tmp, u_data->io->y2_tmp);
1246 u_data->io->y1_tmp = helpy;
1247 u_data->io->x_new=event->x;
1248 u_data->io->y_new=event->y;
1249 u_data->io->rectangle=TRUE;
1250 u_data->io->rectangle_present=TRUE;
1254 if (u_data->io->rectangle_present==TRUE)
1256 u_data->io->rectangle_present=FALSE;
1257 if (event->x >= u_data->io->rect_x_min && event->x <= u_data->io->rect_x_max &&
1258 event->y >= u_data->io->rect_y_min && event->y <= u_data->io->rect_y_max)
1259 zoomin_bt_fcn(u_data);
1262 u_data->io->x1_tmp_sec = u_data->io->x1_akt_sec;
1263 u_data->io->x1_tmp_usec = u_data->io->x1_akt_usec;
1264 u_data->io->x2_tmp_sec = u_data->io->x2_akt_sec;
1265 u_data->io->x2_tmp_usec = u_data->io->x2_akt_usec;
1266 sctp_graph_redraw(u_data);
1272 sctp_graph_redraw(u_data);
1276 x_value = ((event->x-LEFT_BORDER-u_data->io->offset) * ((u_data->io->x2_tmp_sec+u_data->io->x2_tmp_usec/1000000.0)-(u_data->io->x1_tmp_sec+u_data->io->x1_tmp_usec/1000000.0)) / (u_data->io->pixmap_width-LEFT_BORDER-RIGHT_BORDER-u_data->io->offset))+u_data->io->x1_tmp_sec+u_data->io->x1_tmp_usec/1000000.0;
1277 y_value = (gint)rint((u_data->io->pixmap_height-BOTTOM_BORDER-u_data->io->offset-event->y) * (max_tsn - min_tsn) / (u_data->io->pixmap_height-BOTTOM_BORDER-u_data->io->offset)) + min_tsn;
1278 text_color = u_data->io->draw_area->style->black_gc;
1280 if (u_data->dir == 1)
1282 tsnlist = u_data->assoc->sort_tsn1;
1283 t_size = u_data->assoc->n_data_chunks_ep1;
1284 sacklist = u_data->assoc->sort_sack1;
1285 s_size = u_data->assoc->n_sack_chunks_ep1;
1289 tsnlist = u_data->assoc->sort_tsn2;
1290 t_size = u_data->assoc->n_data_chunks_ep2;
1291 sacklist = u_data->assoc->sort_sack2;
1292 s_size = u_data->assoc->n_sack_chunks_ep2;
1294 x_tolerance = (gdouble)((u_data->io->tmp_width / u_data->io->axis_width*1.0))*5/1000000.0;
1295 y_tolerance = (guint32)(((u_data->io->max_y - u_data->io->min_y) / (u_data->io->pixmap_height-TOP_BORDER-BOTTOM_BORDER-u_data->io->offset)) * 2.0);
1298 else if (y_tolerance > 5)
1301 for (i=0; i<s_size; i++)
1303 sack = (struct tsn_sort*)(g_ptr_array_index(sacklist, i));
1304 if ((guint32)abs(sack->tsnumber - y_value)<y_tolerance)
1306 s_diff = fabs((sack->secs+sack->usecs/1000000.0)- x_value);
1307 if (s_diff < x_tolerance)
1313 for (i=0; i<t_size; i++)
1315 tsn = (struct tsn_sort*)(g_ptr_array_index(tsnlist, i));
1316 if ((guint32)abs(tsn->tsnumber - y_value)<y_tolerance)
1318 t_diff = fabs((tsn->secs+tsn->usecs/1000000.0)- x_value);
1319 if (sack_found && s_diff < t_diff)
1321 cf_goto_frame(&cfile, sack->framenumber);
1322 x_value = sack->secs+sack->usecs/1000000.0;
1323 y_value = sack->tsnumber;
1325 else if (t_diff < x_tolerance)
1327 cf_goto_frame(&cfile, tsn->framenumber);
1328 x_value = tsn->secs+tsn->usecs/1000000.0;
1329 y_value = tsn->tsnumber;
1335 g_snprintf(label_string, sizeof(label_string), "(%.6lf, %u)", x_value, y_value);
1339 gdk_draw_line(u_data->io->pixmap,text_color, (gint)(event->x-2), (gint)(event->y), (gint)(event->x+2), (gint)(event->y));
1340 gdk_draw_line(u_data->io->pixmap,text_color, (gint)(event->x), (gint)(event->y-2), (gint)(event->x), (gint)(event->y+2));
1341 if (event->x+150>=u_data->io->pixmap_width)
1342 position = event->x - 150;
1344 position = event->x + 5;
1347 memcpy(label_string,(gchar *)g_locale_to_utf8(label_string, -1 , NULL, NULL, NULL), 15);
1348 pango_layout_set_text(layout, label_string, -1);
1349 pango_layout_get_pixel_size(layout, &lwidth, NULL);
1351 gdk_draw_layout(u_data->io->pixmap,text_color,
1353 (gint)(event->y-10),
1356 ios=(sctp_graph_t *)g_object_get_data(G_OBJECT(u_data->io->draw_area), "sctp_graph_t");
1361 gdk_draw_pixmap(u_data->io->draw_area->window,
1362 u_data->io->draw_area->style->fg_gc[GTK_WIDGET_STATE(u_data->io->draw_area)],
1366 u_data->io->draw_area->allocation.width,
1367 u_data->io->draw_area->allocation.height);
1374 static void init_sctp_graph_window(struct sctp_udata *u_data)
1378 GtkWidget *bt_close, *tsn_bt, *both_bt, *zoomin_bt;
1379 GtkTooltips *tooltip_in, *tooltip_out;
1381 /* create the main window */
1382 u_data->io->window= dlg_window_new("WSCTP Graphics"); /* transient_for top_level */
1383 gtk_window_set_destroy_with_parent (GTK_WINDOW(u_data->io->window), TRUE);
1385 vbox=gtk_vbox_new(FALSE, 0);
1386 gtk_container_add(GTK_CONTAINER(u_data->io->window), vbox);
1387 gtk_widget_show(vbox);
1389 create_draw_area(vbox, u_data);
1391 sctp_graph_set_title(u_data);
1393 hbox = gtk_hbutton_box_new();
1394 gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0);
1395 gtk_container_set_border_width(GTK_CONTAINER(hbox), 10);
1396 gtk_button_box_set_layout(GTK_BUTTON_BOX (hbox), GTK_BUTTONBOX_SPREAD);
1397 gtk_box_set_spacing(GTK_BOX (hbox), 0);
1398 gtk_box_set_child_packing(GTK_BOX(vbox), hbox, FALSE, FALSE, 0, GTK_PACK_START);
1399 gtk_widget_show(hbox);
1400 sack_bt = gtk_button_new_with_label ("Show Only Sacks");
1401 gtk_box_pack_start(GTK_BOX(hbox), sack_bt, FALSE, FALSE, 0);
1402 gtk_widget_show(sack_bt);
1404 g_signal_connect(sack_bt, "clicked", G_CALLBACK(on_sack_bt), u_data);
1405 tsn_bt = gtk_button_new_with_label ("Show Only TSNs");
1406 gtk_box_pack_start(GTK_BOX(hbox), tsn_bt, FALSE, FALSE, 0);
1407 gtk_widget_show(tsn_bt);
1408 g_signal_connect(tsn_bt, "clicked", G_CALLBACK(on_tsn_bt), u_data);
1410 both_bt = gtk_button_new_with_label ("Show both");
1411 gtk_box_pack_start(GTK_BOX(hbox), both_bt, FALSE, FALSE, 0);
1412 gtk_widget_show(both_bt);
1413 g_signal_connect(both_bt, "clicked", G_CALLBACK(on_both_bt), u_data);
1415 zoomin_bt = gtk_button_new_with_label ("Zoom in");
1416 gtk_box_pack_start(GTK_BOX(hbox), zoomin_bt, FALSE, FALSE, 0);
1417 gtk_widget_show(zoomin_bt);
1418 g_signal_connect(zoomin_bt, "clicked", G_CALLBACK(on_zoomin_bt), u_data);
1419 tooltip_in = gtk_tooltips_new();
1420 gtk_tooltips_set_tip(tooltip_in, zoomin_bt, "Zoom in the area you have selected", NULL);
1422 zoomout_bt = gtk_button_new_with_label ("Zoom out");
1423 gtk_box_pack_start(GTK_BOX(hbox), zoomout_bt, FALSE, FALSE, 0);
1424 gtk_widget_show(zoomout_bt);
1425 g_signal_connect(zoomout_bt, "clicked", G_CALLBACK(on_zoomout_bt), u_data);
1426 tooltip_out = gtk_tooltips_new();
1427 gtk_tooltips_set_tip(tooltip_out, zoomout_bt, "Zoom out one step", NULL);
1428 gtk_widget_set_sensitive(zoomout_bt, FALSE);
1430 bt_close = gtk_button_new_from_stock(GTK_STOCK_CLOSE);
1431 gtk_box_pack_start(GTK_BOX(hbox), bt_close, FALSE, FALSE, 0);
1432 gtk_widget_show(bt_close);
1433 g_signal_connect(bt_close, "clicked", G_CALLBACK(sctp_graph_close_cb), u_data);
1435 g_signal_connect(u_data->io->draw_area,"button_press_event",G_CALLBACK(on_button_press), u_data);
1436 g_signal_connect(u_data->io->draw_area,"button_release_event",G_CALLBACK(on_button_release), u_data);
1437 gtk_widget_set_events(u_data->io->draw_area, GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK | GDK_EXPOSURE_MASK);
1439 gtk_widget_show(u_data->io->window);
1442 static void sctp_graph_set_title(struct sctp_udata *u_data)
1446 if(!u_data->io->window)
1450 title = g_strdup_printf("SCTP TSNs and Sacks over Time: %s Port1 %u Port2 %u Endpoint %u",
1451 cf_get_display_name(&cfile), u_data->parent->assoc->port1, u_data->parent->assoc->port2, u_data->dir);
1452 gtk_window_set_title(GTK_WINDOW(u_data->io->window), title);
1457 gtk_sctpgraph_init(struct sctp_udata *u_data)
1461 sctp_min_max_t* tmp_minmax;
1463 io=g_malloc(sizeof(sctp_graph_t));
1464 io->needs_redraw=TRUE;
1465 io->x_interval=1000;
1469 io->pixmap_width=800;
1470 io->pixmap_height=600;
1474 u_data->io->x1_tmp_sec=u_data->assoc->min_secs;
1475 u_data->io->x1_tmp_usec=u_data->assoc->min_usecs;
1476 u_data->io->x2_tmp_sec=u_data->assoc->max_secs;
1477 u_data->io->x2_tmp_usec=u_data->assoc->max_usecs;
1478 u_data->io->tmp_min_tsn1=u_data->assoc->min_tsn1;
1479 u_data->io->tmp_max_tsn1=u_data->assoc->max_tsn1;
1480 u_data->io->tmp_min_tsn2=u_data->assoc->min_tsn2;
1481 u_data->io->tmp_max_tsn2=u_data->assoc->max_tsn2;
1482 u_data->io->tmp=FALSE;
1484 tmp_minmax = g_malloc(sizeof(sctp_min_max_t));
1485 tmp_minmax->tmp_min_secs = u_data->assoc->min_secs;
1486 tmp_minmax->tmp_min_usecs=u_data->assoc->min_usecs;
1487 tmp_minmax->tmp_max_secs=u_data->assoc->max_secs;
1488 tmp_minmax->tmp_max_usecs=u_data->assoc->max_usecs;
1489 tmp_minmax->tmp_min_tsn2=u_data->assoc->min_tsn2;
1490 tmp_minmax->tmp_min_tsn1=u_data->assoc->min_tsn1;
1491 tmp_minmax->tmp_max_tsn1=u_data->assoc->max_tsn1;
1492 tmp_minmax->tmp_max_tsn2=u_data->assoc->max_tsn2;
1493 u_data->assoc->min_max = g_slist_prepend(u_data->assoc->min_max, tmp_minmax);
1496 init_sctp_graph_window(u_data);
1497 sctp_graph_redraw(u_data);
1503 quit(GtkObject *object _U_, gpointer user_data)
1505 struct sctp_udata *u_data=(struct sctp_udata*)user_data;
1507 decrease_childcount(u_data->parent);
1508 remove_child(u_data, u_data->parent);
1512 u_data->assoc->min_max = NULL;
1518 static void create_draw_area(GtkWidget *box, struct sctp_udata *u_data)
1521 u_data->io->draw_area=gtk_drawing_area_new();
1522 g_signal_connect(u_data->io->draw_area, "destroy", G_CALLBACK(quit), u_data);
1523 g_object_set_data(G_OBJECT(u_data->io->draw_area), "sctp_graph_t", u_data->io);
1525 gtk_widget_set_size_request(u_data->io->draw_area, u_data->io->pixmap_width, u_data->io->pixmap_height);
1527 /* signals needed to handle backing pixmap */
1528 g_signal_connect(u_data->io->draw_area, "expose_event", G_CALLBACK(expose_event), NULL);
1529 g_signal_connect(u_data->io->draw_area, "configure_event", G_CALLBACK(configure_event), u_data);
1531 gtk_widget_show(u_data->io->draw_area);
1532 gtk_box_pack_start(GTK_BOX(box), u_data->io->draw_area, TRUE, TRUE, 0);
1537 void create_graph(guint16 dir, struct sctp_analyse* userdata)
1539 struct sctp_udata *u_data;
1541 u_data=g_malloc(sizeof(struct sctp_udata));
1542 u_data->assoc=g_malloc(sizeof(sctp_assoc_info_t));
1543 u_data->assoc=userdata->assoc;
1546 u_data->parent = userdata;
1547 if ((u_data->dir==1 && u_data->assoc->n_array_tsn1==0)|| (u_data->dir==2 && u_data->assoc->n_array_tsn2==0))
1548 simple_dialog(ESD_TYPE_INFO, ESD_BTN_OK, "No Data Chunks sent");
1551 set_child(u_data, u_data->parent);
1552 increase_childcount(u_data->parent);
1553 gtk_sctpgraph_init(u_data);
1557 #if defined(_WIN32) && !defined(__MINGW32__)
1558 /* replacement of Unix rint() for Windows */
1559 static int rint (double x)
1564 buf = _fcvt(x, 0, &dec, &sig);