git-svn-id: http://ltt.polymtl.ca/svn@125 04897980-b3bd-0310-b5e0-8ef037075253
[lttv.git] / ltt / branches / poly / lttv / modules / guiEvents.c
CommitLineData
fbbb2697 1//*! \defgroup GuiEvents libGuiEvents: The GUI Events display plugin */
5c7463ed 2/*\@{*/
3
fbbb2697 4/*! \file GuiEvents.c
5c7463ed 5 * \brief Graphical plugin for showing events.
6 *
17abcce3 7 * This plugin lists all the events contained in the current time interval
8 * in a list.
9 *
5c7463ed 10 * This plugin adds a Events Viewer functionnality to Linux TraceToolkit
11 * GUI when this plugin is loaded. The init and destroy functions add the
12 * viewer's insertion menu item and toolbar icon by calling gtkTraceSet's
13 * API functions. Then, when a viewer's object is created, the constructor
14 * creates ans register through API functions what is needed to interact
15 * with the TraceSet window.
16 *
fbbb2697 17 * Coding standard :
18 * pm : parameter
19 * l : local
20 * g : global
21 * s : static
22 * h : hook
23 *
fcdf0ec2 24 * Author : Karim Yaghmour
25 * Integrated to LTTng by Mathieu Desnoyers, June 2003
5c7463ed 26 */
27
b26121f4 28#include <math.h>
29
5c7463ed 30#include <glib.h>
31#include <gmodule.h>
fbbb2697 32#include <gtk/gtk.h>
33#include <gdk/gdk.h>
5c7463ed 34
35#include <lttv/module.h>
fbbb2697 36//#include <lttv/gtkTraceSet.h>
37#include "mw_api.h"
6601bf27 38#include "gtktreeprivate.h"
5c7463ed 39
fbbb2697 40#include "icons/hGuiEventsInsert.xpm"
5c7463ed 41
fcdf0ec2 42/** Array containing instanced objects. Used when module is unloaded */
fbbb2697 43static GSList *sEvent_Viewer_Data_List = NULL ;
44
45typedef struct _EventViewerData {
46
47 /* Model containing list data */
48 GtkListStore *Store_M;
49
50 GtkWidget *HBox_V;
51 /* Widget to display the data in a columned list */
52 GtkWidget *Tree_V;
53 GtkAdjustment *VTree_Adjust_C ;
54 GdkWindow *TreeWindow ;
b26121f4 55
fbbb2697 56 /* Vertical scrollbar and it's adjustment */
57 GtkWidget *VScroll_VC;
58 GtkAdjustment *VAdjust_C ;
59
60 /* Selection handler */
61 GtkTreeSelection *Select_C;
62
b26121f4 63 guint Num_Visible_Events;
64 guint First_Event, Last_Event;
fcdf0ec2 65
fbbb2697 66} EventViewerData ;
67
68//! Event Viewer's constructor hook
69GtkWidget *hGuiEvents(GtkWidget *pmParentWindow);
5c7463ed 70//! Event Viewer's constructor
fbbb2697 71EventViewerData *GuiEvents(void);
72//! Event Viewer's destructor
73void GuiEvents_Destructor(EventViewerData *Event_Viewer_Data);
74
75/* Prototype for selection handler callback */
76static void tree_selection_changed_cb (GtkTreeSelection *selection, gpointer data);
77static void v_scroll_cb (GtkAdjustment *adjustment, gpointer data);
78static void Tree_V_size_allocate_cb (GtkWidget *widget, GtkAllocation *alloc, gpointer data);
79static void Tree_V_size_request_cb (GtkWidget *widget, GtkRequisition *requisition, gpointer data);
b26121f4 80static void Tree_V_cursor_changed_cb (GtkWidget *widget, gpointer data);
81static void Tree_V_move_cursor_cb (GtkWidget *widget, GtkMovementStep arg1, gint arg2, gpointer data);
fbbb2697 82
b26121f4 83static void get_test_data(guint Event_Number, guint List_Height,
84 EventViewerData *Event_Viewer_Data);
fbbb2697 85
86void add_test_data(EventViewerData *Event_Viewer_Data);
87
88/* TEST DATA, TO BE READ FROM THE TRACE */
89static int Number_Of_Events = 1000;
90
5c7463ed 91/**
92 * plugin's init function
93 *
94 * This function initializes the Event Viewer functionnality through the
95 * gtkTraceSet API.
96 */
97G_MODULE_EXPORT void init() {
98 g_critical("GUI Event Viewer init()");
99
100 /* Register the toolbar insert button */
fbbb2697 101 //ToolbarItemReg(hGuiEventsInsert_xpm, "Insert Event Viewer", hGuiEvents);
5c7463ed 102
103 /* Register the menu item insert entry */
fbbb2697 104 //MenuItemReg("/", "Insert Event Viewer", hGuiEvents);
105}
5c7463ed 106
fbbb2697 107void destroy_walk(gpointer data, gpointer user_data)
108{
109 GuiEvents_Destructor((EventViewerData*)data);
5c7463ed 110}
111
112/**
113 * plugin's destroy function
114 *
115 * This function releases the memory reserved by the module and unregisters
116 * everything that has been registered in the gtkTraceSet API.
117 */
118G_MODULE_EXPORT void destroy() {
fcdf0ec2 119 int i;
fbbb2697 120
121 EventViewerData *Event_Viewer_Data;
fcdf0ec2 122
5c7463ed 123 g_critical("GUI Event Viewer destroy()");
124
fbbb2697 125 g_slist_foreach(sEvent_Viewer_Data_List, destroy_walk, NULL );
fcdf0ec2 126
5c7463ed 127 /* Unregister the toolbar insert button */
fbbb2697 128 //ToolbarItemUnreg(hGuiEvents);
5c7463ed 129
130 /* Unregister the menu item insert entry */
fbbb2697 131 //MenuItemUnreg(hGuiEvents);
5c7463ed 132}
133
fbbb2697 134/* Enumeration of the columns */
135enum
136{
137 CPUID_COLUMN,
138 EVENT_COLUMN,
139 TIME_COLUMN,
140 PID_COLUMN,
141 ENTRY_LEN_COLUMN,
142 EVENT_DESCR_COLUMN,
143 N_COLUMNS
144};
145
146
5c7463ed 147/**
fbbb2697 148 * Event Viewer's constructor hook
5c7463ed 149 *
150 * This constructor is given as a parameter to the menuitem and toolbar button
fbbb2697 151 * registration. It creates the list.
152 * @param pmParentWindow A pointer to the parent window.
5c7463ed 153 * @return The widget created.
154 */
fbbb2697 155GtkWidget *
156hGuiEvents(GtkWidget *pmParentWindow)
157{
158 EventViewerData* Event_Viewer_Data = GuiEvents() ;
159
160 return Event_Viewer_Data->HBox_V ;
161
162}
163
164/**
165 * Event Viewer's constructor
166 *
167 * This constructor is used to create EventViewerData data structure.
168 * @return The Event viewer data created.
169 */
170EventViewerData *
171GuiEvents(void)
5c7463ed 172{
fbbb2697 173 GtkTreeViewColumn *column;
174 GtkCellRenderer *renderer;
175 EventViewerData* Event_Viewer_Data = g_new(EventViewerData,1) ;
b26121f4 176
177
fbbb2697 178 /* Create a model for storing the data list */
179 Event_Viewer_Data->Store_M = gtk_list_store_new (N_COLUMNS, /* Total number of columns */
180 G_TYPE_INT, /* CPUID */
181 G_TYPE_STRING, /* Event */
182 G_TYPE_INT, /* Time */
183 G_TYPE_INT, /* PID */
184 G_TYPE_INT, /* Entry length */
185 G_TYPE_STRING); /* Event's description */
186
187 /* Create the viewer widget for the columned list */
188 Event_Viewer_Data->Tree_V = gtk_tree_view_new_with_model (GTK_TREE_MODEL (Event_Viewer_Data->Store_M));
b26121f4 189
fbbb2697 190 g_signal_connect (G_OBJECT (Event_Viewer_Data->Tree_V), "size-allocate",
191 G_CALLBACK (Tree_V_size_allocate_cb),
192 Event_Viewer_Data);
193 g_signal_connect (G_OBJECT (Event_Viewer_Data->Tree_V), "size-request",
194 G_CALLBACK (Tree_V_size_request_cb),
195 Event_Viewer_Data);
196
b26121f4 197 g_signal_connect (G_OBJECT (Event_Viewer_Data->Tree_V), "cursor-changed",
198 G_CALLBACK (Tree_V_cursor_changed_cb),
199 Event_Viewer_Data);
fbbb2697 200
b26121f4 201 g_signal_connect (G_OBJECT (Event_Viewer_Data->Tree_V), "move-cursor",
202 G_CALLBACK (Tree_V_move_cursor_cb),
203 Event_Viewer_Data);
fbbb2697 204// Use on each column!
205//gtk_tree_view_column_set_sizing(Event_Viewer_Data->Tree_V, GTK_TREE_VIEW_COLUMN_FIXED);
206
207 /* The view now holds a reference. We can get rid of our own
208 * reference */
209 g_object_unref (G_OBJECT (Event_Viewer_Data->Store_M));
fcdf0ec2 210
211
fbbb2697 212 /* Create a column, associating the "text" attribute of the
213 * cell_renderer to the first column of the model */
214 /* Columns alignment : 0.0 : Left 0.5 : Center 1.0 : Right */
215 renderer = gtk_cell_renderer_text_new ();
216 column = gtk_tree_view_column_new_with_attributes ("CPUID",
217 renderer,
218 "text", CPUID_COLUMN,
219 NULL);
220 gtk_tree_view_column_set_alignment (column, 0.0);
221 gtk_tree_view_column_set_fixed_width (column, 45);
222 gtk_tree_view_append_column (GTK_TREE_VIEW (Event_Viewer_Data->Tree_V), column);
223
224 renderer = gtk_cell_renderer_text_new ();
225 column = gtk_tree_view_column_new_with_attributes ("Event",
226 renderer,
227 "text", EVENT_COLUMN,
228 NULL);
229 gtk_tree_view_column_set_alignment (column, 0.0);
230 gtk_tree_view_column_set_fixed_width (column, 120);
231 gtk_tree_view_append_column (GTK_TREE_VIEW (Event_Viewer_Data->Tree_V), column);
232
233 renderer = gtk_cell_renderer_text_new ();
234 column = gtk_tree_view_column_new_with_attributes ("Time",
235 renderer,
236 "text", TIME_COLUMN,
237 NULL);
238 gtk_tree_view_column_set_alignment (column, 1.0);
239 gtk_tree_view_column_set_fixed_width (column, 120);
240 gtk_tree_view_append_column (GTK_TREE_VIEW (Event_Viewer_Data->Tree_V), column);
241
242 renderer = gtk_cell_renderer_text_new ();
243 column = gtk_tree_view_column_new_with_attributes ("PID",
244 renderer,
245 "text", PID_COLUMN,
246 NULL);
247 gtk_tree_view_column_set_alignment (column, 1.0);
248 gtk_tree_view_column_set_fixed_width (column, 45);
249 gtk_tree_view_append_column (GTK_TREE_VIEW (Event_Viewer_Data->Tree_V), column);
250
251 renderer = gtk_cell_renderer_text_new ();
252 column = gtk_tree_view_column_new_with_attributes ("Entry Length",
253 renderer,
254 "text", ENTRY_LEN_COLUMN,
255 NULL);
256 gtk_tree_view_column_set_alignment (column, 1.0);
257 gtk_tree_view_column_set_fixed_width (column, 60);
258 gtk_tree_view_append_column (GTK_TREE_VIEW (Event_Viewer_Data->Tree_V), column);
259
260 renderer = gtk_cell_renderer_text_new ();
261 column = gtk_tree_view_column_new_with_attributes ("Event's Description",
262 renderer,
263 "text", EVENT_DESCR_COLUMN,
264 NULL);
265 gtk_tree_view_column_set_alignment (column, 0.0);
266 gtk_tree_view_append_column (GTK_TREE_VIEW (Event_Viewer_Data->Tree_V), column);
267
268
fbbb2697 269 /* Setup the selection handler */
270 Event_Viewer_Data->Select_C = gtk_tree_view_get_selection (GTK_TREE_VIEW (Event_Viewer_Data->Tree_V));
271 gtk_tree_selection_set_mode (Event_Viewer_Data->Select_C, GTK_SELECTION_SINGLE);
272 g_signal_connect (G_OBJECT (Event_Viewer_Data->Select_C), "changed",
273 G_CALLBACK (tree_selection_changed_cb),
b26121f4 274 Event_Viewer_Data);
fbbb2697 275
276 Event_Viewer_Data->HBox_V = gtk_hbox_new(0, 0);
277 gtk_box_pack_start(GTK_BOX(Event_Viewer_Data->HBox_V), Event_Viewer_Data->Tree_V, TRUE, TRUE, 0);
fcdf0ec2 278
279 /* Create vertical scrollbar and pack it */
fbbb2697 280 Event_Viewer_Data->VScroll_VC = gtk_vscrollbar_new(NULL);
281 gtk_box_pack_start(GTK_BOX(Event_Viewer_Data->HBox_V), Event_Viewer_Data->VScroll_VC, FALSE, TRUE, 0);
282
fcdf0ec2 283 /* Get the vertical scrollbar's adjustment */
fbbb2697 284 Event_Viewer_Data->VAdjust_C = gtk_range_get_adjustment(GTK_RANGE(Event_Viewer_Data->VScroll_VC));
285 Event_Viewer_Data->VTree_Adjust_C = gtk_tree_view_get_vadjustment(
286 GTK_TREE_VIEW (Event_Viewer_Data->Tree_V));
287
288 g_signal_connect (G_OBJECT (Event_Viewer_Data->VAdjust_C), "value-changed",
289 G_CALLBACK (v_scroll_cb),
290 Event_Viewer_Data);
291 /* Set the upper bound to the last event number */
292 Event_Viewer_Data->VAdjust_C->lower = 0;
293 Event_Viewer_Data->VAdjust_C->upper = Number_Of_Events;
294 Event_Viewer_Data->VAdjust_C->value = 0;
295 Event_Viewer_Data->VAdjust_C->step_increment = 1;
296 Event_Viewer_Data->VAdjust_C->page_increment =
297 Event_Viewer_Data->VTree_Adjust_C->upper;
fbbb2697 298 Event_Viewer_Data->VAdjust_C->page_size =
299 Event_Viewer_Data->VTree_Adjust_C->upper;
300 g_critical("value : %u",Event_Viewer_Data->VTree_Adjust_C->upper);
301 /* Raw event trace */
302 gtk_widget_show(Event_Viewer_Data->HBox_V);
303 gtk_widget_show(Event_Viewer_Data->Tree_V);
304 gtk_widget_show(Event_Viewer_Data->VScroll_VC);
305
306 /* Add the object's information to the module's array */
307 g_slist_append(sEvent_Viewer_Data_List, Event_Viewer_Data);
308
b26121f4 309 Event_Viewer_Data->First_Event = -1 ;
310 Event_Viewer_Data->Last_Event = 0 ;
311
312 Event_Viewer_Data->Num_Visible_Events = 1;
313
fbbb2697 314 /* Test data */
b26121f4 315 get_test_data((int)Event_Viewer_Data->VAdjust_C->value,
316 Event_Viewer_Data->Num_Visible_Events,
317 Event_Viewer_Data);
fbbb2697 318
319 return Event_Viewer_Data;
320}
fcdf0ec2 321
b26121f4 322void Tree_V_move_cursor_cb (GtkWidget *widget, GtkMovementStep arg1, gint arg2, gpointer data)
fbbb2697 323{
b26121f4 324 GtkTreePath *path; // = gtk_tree_path_new();
325 gint *indices;
326 gdouble value;
327 EventViewerData *Event_Viewer_Data = (EventViewerData*)data;
328
329 gtk_tree_view_get_cursor(GTK_TREE_VIEW(Event_Viewer_Data->Tree_V), &path, NULL);
330 if(path == NULL)
331 {
332 /* No prior cursor, put it at beginning of page and let the execution do */
333 path = gtk_tree_path_new_from_indices(0, -1);
334 gtk_tree_view_set_cursor(GTK_TREE_VIEW(Event_Viewer_Data->Tree_V), path, NULL, FALSE);
335 }
336
337 indices = gtk_tree_path_get_indices(path);
338
339 g_critical("DEBUG : move cursor step : %u , int : %i , indice : %i", (guint)arg1, arg2, indices[0]) ;
340
341 value = gtk_adjustment_get_value(Event_Viewer_Data->VAdjust_C);
342
343 if(arg1 == GTK_MOVEMENT_DISPLAY_LINES)
344 {
345 /* Move one line */
346 if(arg2 == 1)
347 {
348 /* move one line down */
349 if(indices[0] == Event_Viewer_Data->Num_Visible_Events - 1)
350 {
351 if(value + Event_Viewer_Data->Num_Visible_Events <= Number_Of_Events -1)
352 {
353 g_critical("need 1 event down") ;
354 gtk_adjustment_set_value(Event_Viewer_Data->VAdjust_C, value+1);
355 gtk_tree_path_free(path);
356 path = gtk_tree_path_new_from_indices(Event_Viewer_Data->Num_Visible_Events-1, -1);
357 gtk_tree_view_set_cursor(GTK_TREE_VIEW(Event_Viewer_Data->Tree_V), path, NULL, FALSE);
358 g_signal_stop_emission_by_name(G_OBJECT(widget), "move-cursor");
359 }
360 }
361 } else {
362 /* Move one line up */
363 if(indices[0] == 0)
364 {
365 if(value - 1 >= 0 )
366 {
367 g_critical("need 1 event up") ;
368 gtk_adjustment_set_value(Event_Viewer_Data->VAdjust_C, value-1);
369 gtk_tree_path_free(path);
370 path = gtk_tree_path_new_from_indices(0, -1);
371 gtk_tree_view_set_cursor(GTK_TREE_VIEW(Event_Viewer_Data->Tree_V), path, NULL, FALSE);
372 g_signal_stop_emission_by_name(G_OBJECT(widget), "move-cursor");
373 }
374
375 }
376 }
377
378 }
379
380 if(arg1 == GTK_MOVEMENT_PAGES)
381 {
382 /* Move one page */
383 if(arg2 == 1)
384 {
385 if(Event_Viewer_Data->Num_Visible_Events == 1)
386 value += 1 ;
387 /* move one page down */
388 if(value + Event_Viewer_Data->Num_Visible_Events-1 <= Number_Of_Events )
389 {
390 g_critical("need 1 page down") ;
391
392 gtk_adjustment_set_value(Event_Viewer_Data->VAdjust_C,
393 value+(Event_Viewer_Data->Num_Visible_Events-1));
394 gtk_tree_path_free(path);
395 path = gtk_tree_path_new_from_indices(0, -1);
396 gtk_tree_view_set_cursor(GTK_TREE_VIEW(Event_Viewer_Data->Tree_V), path, NULL, FALSE);
397 g_signal_stop_emission_by_name(G_OBJECT(widget), "move-cursor");
398 }
399 } else {
400 /* Move one page up */
401 if(Event_Viewer_Data->Num_Visible_Events == 1)
402 value -= 1 ;
403
404 if(indices[0] < Event_Viewer_Data->Num_Visible_Events - 2 )
405 {
406 if(value - (Event_Viewer_Data->Num_Visible_Events-1) >= 0)
407 {
408 g_critical("need 1 page up") ;
409
410
411 gtk_adjustment_set_value(Event_Viewer_Data->VAdjust_C,
412 value-(Event_Viewer_Data->Num_Visible_Events-1));
413 gtk_tree_path_free(path);
414 path = gtk_tree_path_new_from_indices(0, -1);
415 gtk_tree_view_set_cursor(GTK_TREE_VIEW(Event_Viewer_Data->Tree_V), path, NULL, FALSE);
416 g_signal_stop_emission_by_name(G_OBJECT(widget), "move-cursor");
417
418 }
419 }
420 }
421
422 }
423
424 if(arg1 == GTK_MOVEMENT_BUFFER_ENDS)
425 {
426 /* Move to the ends of the buffer */
427 if(arg2 == 1)
428 {
429 /* move end of buffer */
430 g_critical("End of buffer") ;
431 gtk_adjustment_set_value(Event_Viewer_Data->VAdjust_C, Number_Of_Events-Event_Viewer_Data->Num_Visible_Events);
432 gtk_tree_path_free(path);
433 path = gtk_tree_path_new_from_indices(Event_Viewer_Data->Num_Visible_Events-1, -1);
434 gtk_tree_view_set_cursor(GTK_TREE_VIEW(Event_Viewer_Data->Tree_V), path, NULL, FALSE);
435 g_signal_stop_emission_by_name(G_OBJECT(widget), "move-cursor");
436 } else {
437 /* Move beginning of buffer */
438 g_critical("Beginning of buffer") ;
439 gtk_adjustment_set_value(Event_Viewer_Data->VAdjust_C, 0);
440 gtk_tree_path_free(path);
441 path = gtk_tree_path_new_from_indices(0, -1);
442 gtk_tree_view_set_cursor(GTK_TREE_VIEW(Event_Viewer_Data->Tree_V), path, NULL, FALSE);
443 g_signal_stop_emission_by_name(G_OBJECT(widget), "move-cursor");
444 }
445
446 }
447
fbbb2697 448
b26121f4 449
450 gtk_tree_path_free(path);
451
452}
453
454void Tree_V_cursor_changed_cb (GtkWidget *widget, gpointer data)
455{
456 g_critical("DEBUG : cursor change");
457
458}
459
460
461void v_scroll_cb (GtkAdjustment *adjustment, gpointer data)
462{
463 EventViewerData *Event_Viewer_Data = (EventViewerData*)data;
464 g_critical("DEBUG : scroll signal, value : %f", adjustment->value);
465
466 get_test_data((int)adjustment->value, Event_Viewer_Data->Num_Visible_Events,
467 Event_Viewer_Data);
468
fbbb2697 469}
fcdf0ec2 470
6601bf27 471gint get_cell_height(GtkTreeView *TreeView)
472{
473 gint height, width;
474 GtkTreeViewColumn *Column = gtk_tree_view_get_column(TreeView, 0);
475 GList *Render_List = gtk_tree_view_column_get_cell_renderers(Column);
476 GtkCellRenderer *Renderer = g_list_first(Render_List)->data;
477
b26121f4 478 gtk_tree_view_column_cell_get_size(Column, NULL, NULL, NULL, NULL, &height);
479 g_critical("cell 0 height : %u",height);
6601bf27 480
481 return height;
482}
483
fbbb2697 484void Tree_V_size_allocate_cb (GtkWidget *widget, GtkAllocation *alloc, gpointer data)
485{
486 EventViewerData *Event_Viewer_Data = (EventViewerData*)data;
6601bf27 487 gint Cell_Height = get_cell_height(GTK_TREE_VIEW(Event_Viewer_Data->Tree_V));
b26121f4 488 gint Last_Num_Visible_Events = Event_Viewer_Data->Num_Visible_Events;
489 gdouble Exact_Num_Visible;
490
fbbb2697 491 g_critical("size-allocate");
fcdf0ec2 492
b26121f4 493 Exact_Num_Visible = ( alloc->height -
6601bf27 494 TREE_VIEW_HEADER_HEIGHT (GTK_TREE_VIEW(Event_Viewer_Data->Tree_V)) )
b26121f4 495 / (double)Cell_Height ;
496
497 Event_Viewer_Data->Num_Visible_Events = ceil(Exact_Num_Visible) ;
498
499 g_critical("number of events shown : %u",Event_Viewer_Data->Num_Visible_Events);
500 g_critical("ex number of events shown : %f",Exact_Num_Visible);
6601bf27 501
502 Event_Viewer_Data->VAdjust_C->page_increment =
b26121f4 503 floor(Exact_Num_Visible);
6601bf27 504 Event_Viewer_Data->VAdjust_C->page_size =
b26121f4 505 floor(Exact_Num_Visible);
506
507 if(Event_Viewer_Data->Num_Visible_Events != Last_Num_Visible_Events)
508 {
509 get_test_data((int)Event_Viewer_Data->VAdjust_C->value,
510 Event_Viewer_Data->Num_Visible_Events,
511 Event_Viewer_Data);
512 }
513
6601bf27 514
fbbb2697 515}
516
517void Tree_V_size_request_cb (GtkWidget *widget, GtkRequisition *requisition, gpointer data)
518{
6601bf27 519 gint h;
fbbb2697 520 EventViewerData *Event_Viewer_Data = (EventViewerData*)data;
6601bf27 521 gint Cell_Height = get_cell_height(GTK_TREE_VIEW(Event_Viewer_Data->Tree_V));
fbbb2697 522
523 g_critical("size-request");
524
6601bf27 525 h = Cell_Height + TREE_VIEW_HEADER_HEIGHT
526 (GTK_TREE_VIEW(Event_Viewer_Data->Tree_V));
527 requisition->height = h;
fbbb2697 528
fcdf0ec2 529}
530
b26121f4 531void get_test_data(guint Event_Number, guint List_Height,
532 EventViewerData *Event_Viewer_Data)
533{
534 GtkTreeIter iter;
535 int i;
536 GtkTreeModel *model = GTK_TREE_MODEL(Event_Viewer_Data->Store_M);
537 GtkTreePath *Tree_Path;
538 gchar *test_string;
539
540// if(Event_Number > Event_Viewer_Data->Last_Event ||
541// Event_Number + List_Height-1 < Event_Viewer_Data->First_Event ||
542// Event_Viewer_Data->First_Event == -1)
543 {
544 /* no event can be reused, clear and start from nothing */
545 gtk_list_store_clear(Event_Viewer_Data->Store_M);
546 for(i=Event_Number; i<Event_Number+List_Height; i++)
547 {
548 if(i>=Number_Of_Events) break;
549 /* Add a new row to the model */
550 gtk_list_store_append (Event_Viewer_Data->Store_M, &iter);
551 gtk_list_store_set (Event_Viewer_Data->Store_M, &iter,
552 CPUID_COLUMN, 0,
553 EVENT_COLUMN, "event irq",
554 TIME_COLUMN, i,
555 PID_COLUMN, 100,
556 ENTRY_LEN_COLUMN, 17,
557 EVENT_DESCR_COLUMN, "Detailed information",
558 -1);
559 }
560 }
561#ifdef DEBUG //do not use this, it's slower and broken
562// } else {
563 /* Some events will be reused */
564 if(Event_Number < Event_Viewer_Data->First_Event)
565 {
566 /* scrolling up, prepend events */
567 Tree_Path = gtk_tree_path_new_from_indices
568 (Event_Number+List_Height-1 -
569 Event_Viewer_Data->First_Event + 1,
570 -1);
571 for(i=0; i<Event_Viewer_Data->Last_Event-(Event_Number+List_Height-1);
572 i++)
573 {
574 /* Remove the last events from the list */
575 if(gtk_tree_model_get_iter(model, &iter, Tree_Path))
576 gtk_list_store_remove(Event_Viewer_Data->Store_M, &iter);
577 }
578
579 for(i=Event_Viewer_Data->First_Event-1; i>=Event_Number; i--)
580 {
581 if(i>=Number_Of_Events) break;
582 /* Prepend new events */
583 gtk_list_store_prepend (Event_Viewer_Data->Store_M, &iter);
584 gtk_list_store_set (Event_Viewer_Data->Store_M, &iter,
585 CPUID_COLUMN, 0,
586 EVENT_COLUMN, "event irq",
587 TIME_COLUMN, i,
588 PID_COLUMN, 100,
589 ENTRY_LEN_COLUMN, 17,
590 EVENT_DESCR_COLUMN, "Detailed information",
591 -1);
592 }
593 } else {
594 /* Scrolling down, append events */
595 for(i=Event_Viewer_Data->First_Event; i<Event_Number; i++)
596 {
597 /* Remove these events from the list */
598 gtk_tree_model_get_iter_first(model, &iter);
599 gtk_list_store_remove(Event_Viewer_Data->Store_M, &iter);
600 }
601 for(i=Event_Viewer_Data->Last_Event+1; i<Event_Number+List_Height; i++)
602 {
603 if(i>=Number_Of_Events) break;
604 /* Append new events */
605 gtk_list_store_append (Event_Viewer_Data->Store_M, &iter);
606 gtk_list_store_set (Event_Viewer_Data->Store_M, &iter,
607 CPUID_COLUMN, 0,
608 EVENT_COLUMN, "event irq",
609 TIME_COLUMN, i,
610 PID_COLUMN, 100,
611 ENTRY_LEN_COLUMN, 17,
612 EVENT_DESCR_COLUMN, "Detailed information",
613 -1);
614 }
615
616 }
617 }
618#endif //DEBUG
619 Event_Viewer_Data->First_Event = Event_Number ;
620 Event_Viewer_Data->Last_Event = Event_Number+List_Height-1 ;
621
622}
623
fbbb2697 624
625void add_test_data(EventViewerData *Event_Viewer_Data)
fcdf0ec2 626{
fbbb2697 627 GtkTreeIter iter;
628 int i;
629
630 for(i=0; i<10; i++)
631 {
632 /* Add a new row to the model */
633 gtk_list_store_append (Event_Viewer_Data->Store_M, &iter);
634 gtk_list_store_set (Event_Viewer_Data->Store_M, &iter,
635 CPUID_COLUMN, 0,
636 EVENT_COLUMN, "event irq",
637 TIME_COLUMN, i,
638 PID_COLUMN, 100,
639 ENTRY_LEN_COLUMN, 17,
640 EVENT_DESCR_COLUMN, "Detailed information",
641 -1);
642 }
643
644}
645
fcdf0ec2 646
fbbb2697 647void
648GuiEvents_Destructor(EventViewerData *Event_Viewer_Data)
649{
650 guint index;
651
652 /* May already been done by GTK window closing */
653 if(GTK_IS_WIDGET(Event_Viewer_Data->HBox_V))
654 gtk_widget_destroy(Event_Viewer_Data->HBox_V);
655
656 /* Destroy the Tree View */
657 //gtk_widget_destroy(Event_Viewer_Data->Tree_V);
658
659 /* Clear raw event list */
660 //gtk_list_store_clear(Event_Viewer_Data->Store_M);
661 //gtk_widget_destroy(GTK_WIDGET(Event_Viewer_Data->Store_M));
662
663 g_slist_remove(sEvent_Viewer_Data_List,Event_Viewer_Data);
fcdf0ec2 664}
665
fbbb2697 666//FIXME : call hGuiEvents_Destructor for corresponding data upon widget destroy
667
668static void
669tree_selection_changed_cb (GtkTreeSelection *selection, gpointer data)
670{
b26121f4 671 EventViewerData *Event_Viewer_Data = (EventViewerData*)data;
fbbb2697 672 GtkTreeIter iter;
b26121f4 673 GtkTreeModel *model = GTK_TREE_MODEL(Event_Viewer_Data->Store_M);
fbbb2697 674 gchar *Event;
675
676 if (gtk_tree_selection_get_selected (selection, &model, &iter))
677 {
678 gtk_tree_model_get (model, &iter, EVENT_COLUMN, &Event, -1);
679
680 g_print ("Event selected : %s\n", Event);
681
682 g_free (Event);
683 }
684}
fcdf0ec2 685
fcdf0ec2 686
687
fbbb2697 688
689/* Imported code from LTT 0.9.6pre2 tracevisualizer */
690#ifdef DEBUG
691
fcdf0ec2 692/******************************************************************
693 * Function :
694 * WDI_gtk_clist_set_last_row_data_full()
695 * Description :
696 * Appends data to the last row of a GtkClist.
697 * Parameters :
698 * Return values :
699 * NONE.
700 * History :
701 * J.H.D., 27/08/99, Initial typing.
702 * Note :
703 * Based on gtk_clist_set_row_data_full() version 1.2.3.
704 * Much faster than using gtk_clist_set_row_data_full().
705 ******************************************************************/
fbbb2697 706static void WDI_gtk_clist_set_last_row_data_full(GtkCList* pmClist,
fcdf0ec2 707 gpointer pmData,
708 GtkDestroyNotify pmDestroy)
709{
710 GtkCListRow *pClistRow;
5c7463ed 711
fcdf0ec2 712 g_return_if_fail (pmClist != NULL);
713 g_return_if_fail (GTK_IS_CLIST (pmClist));
714 g_return_if_fail (pmClist->row_list_end != NULL);
5c7463ed 715
fcdf0ec2 716 pClistRow = pmClist->row_list_end->data;
717 pClistRow->data = pmData;
718 pClistRow->destroy = pmDestroy;
5c7463ed 719}
720
721
fcdf0ec2 722/******************************************************************
723 * Function :
724 * SHRTEventSelect()
725 * Description :
726 * Parameters :
727 * Return values :
728 * History :
729 * Note :
730 ******************************************************************/
fbbb2697 731static void SHRTEventSelect(GtkWidget* pmCList,
fcdf0ec2 732 gint pmRow,
733 gint pmColumn,
734 GdkEventButton* pmEvent,
735 gpointer pmData)
736{
737 systemView* pSysView; /* The system being displayed */
738
739 /* Do we have anything meaningfull */
740 if((pSysView = (systemView*) pmData) == NULL)
741 return;
742
743 /* Store the selected event */
744 pSysView->Window->LastSelectedEvent = *(event*) gtk_clist_get_row_data(GTK_CLIST(pmCList), pmRow);
745 pSysView->Window->EventSelected = TRUE;
746}
747
748/******************************************************************
749 * Function :
750 * SHRTEventButtonPress()
751 * Description :
752 * Parameters :
753 * Return values :
754 * History :
755 * Note :
756 ******************************************************************/
fbbb2697 757static void SHRTEventButtonPress(GtkWidget* pmCList,
fcdf0ec2 758 GdkEventButton* pmEvent,
759 gpointer pmData)
760{
761 systemView* pSysView; /* The system being displayed */
762 gint row, column; /* The clicked row and column */
763
764 /* Do we have anything meaningfull */
765 if((pSysView = (systemView*) pmData) == NULL)
766 return;
767
768 /* if we have a right-click event */
769 if(pmEvent->button == 3)
770 /* If we clicked on an item, get its row and column values */
771 if(gtk_clist_get_selection_info(GTK_CLIST(pmCList), pmEvent->x, pmEvent->y, &row, &column))
772 {
773 /* Highlight the selected row */
774 gtk_clist_select_row(GTK_CLIST(pmCList), row, column);
775
776 /* Store the selected event */
777 pSysView->Window->LastSelectedEvent = *(event*) gtk_clist_get_row_data(GTK_CLIST(pmCList), row);
778 pSysView->Window->EventSelected = TRUE;
779
780 /* Display the popup menu */
781 gtk_menu_popup(GTK_MENU(pSysView->Window->RawEventPopup),
782 NULL, NULL, NULL, NULL,
783 pmEvent->button, GDK_CURRENT_TIME);
784 }
785}
786
787
788/******************************************************************
789 * Function :
790 * SHRTVAdjustValueChanged()
791 * Description :
792 * Parameters :
793 * Return values :
794 * History :
795 * Note :
796 ******************************************************************/
fbbb2697 797static void SHRTVAdjustValueChanged(GtkAdjustment* pmVAdjust,
fcdf0ec2 798 gpointer pmData)
799{
800 event lEvent; /* Event used for searching */
801 guint32 lPosition; /* The position to scroll to */
802 systemView* pSysView; /* The system being displayed */
803
804 /* Do we have anything meaningfull */
805 if((pSysView = (systemView*) pmData) == NULL)
806 return;
807
808 /* Is there an event database? */
809 if(pSysView->EventDB == NULL)
810 return;
811
812 /* Set the pointer to the first event */
813 if(pSysView->EventDB->TraceStart == NULL)
814 return;
815
816 /* Are we closer to the beginning? */
817 if((pmVAdjust->value - (pmVAdjust->upper / 2)) < 0)
818 {
819 /* Set the navigation pointer to the beginning of the list */
820 lEvent = pSysView->EventDB->FirstEvent;
821
822 /* Calculate distance from beginning */
823 lPosition = (guint32) pmVAdjust->value;
824
825 /* Find the event in the event database */
826 while(lPosition > 0)
827 {
828 lPosition--;
829 if(DBEventNext(pSysView->EventDB, &lEvent) != TRUE)
830 break;
831 }
832 }
833 else
834 {
835 /* Set the navigation pointer to the end of the list */
836 lEvent = pSysView->EventDB->LastEvent;
837
838 /* Calculate distance from end */
839 lPosition = (guint32) (pmVAdjust->upper - pmVAdjust->value);
840
841 /* Find the event in the event database */
842 while(lPosition > 0)
843 {
844 lPosition--;
845 if(DBEventPrev(pSysView->EventDB, &lEvent) != TRUE)
846 break;
847 }
848 }
849
850 /* Fill the event list according to what was found */
851 WDFillEventList(pSysView->Window->RTCList,
852 pSysView->EventDB,
853 pSysView->System,
854 &lEvent,
855 &(pSysView->Window->LastSelectedEvent));
856}
857
858
859
860/******************************************************************
861 * Function :
862 * WDConnectSignals()
863 * Description :
864 * Attaches signal handlers to the window items.
865 * Parameters :
866 * pmSysView, System view for which signals have to be connected
867 * Return values :
868 * NONE
869 * History :
870 * Note :
871 * This function attaches a pointer to the main window during
872 * the connect. This means that the handlers will get a pointer
873 * to the window in the data argument.
874 ******************************************************************/
fbbb2697 875static void WDConnectSignals(systemView* pmSysView)
fcdf0ec2 876{
877 /* Raw event Popup menu */
878 gtk_signal_connect(GTK_OBJECT(pmSysView->Window->RawGotoProcess),
879 "activate",
880 GTK_SIGNAL_FUNC(SHGotoProcAnalysis),
881 pmSysView);
882 gtk_signal_connect(GTK_OBJECT(pmSysView->Window->RawViewEvent),
883 "activate",
884 GTK_SIGNAL_FUNC(SHViewEventInEG),
885 pmSysView);
886
887 /* Set event list callbacks */
888 gtk_signal_connect(GTK_OBJECT(pmSysView->Window->RTCList),
889 "select_row",
890 GTK_SIGNAL_FUNC(SHRTEventSelect),
891 pmSysView);
892 gtk_signal_connect(GTK_OBJECT(pmSysView->Window->RTCList),
893 "button-press-event",
894 GTK_SIGNAL_FUNC(SHRTEventButtonPress),
895 pmSysView);
896 gtk_signal_connect(GTK_OBJECT(pmSysView->Window->RTVAdjust),
897 "value-changed",
898 GTK_SIGNAL_FUNC(SHRTVAdjustValueChanged),
899 pmSysView);
900
901
902}
903
904
905/******************************************************************
906 * Function :
907 * WDFillEventList()
908 * Description :
909 * Fills the window's event list using the trace database.
910 * Parameters :
911 * pmList, The list to be filled.
912 * pmTraceDB, The database of events.
913 * pmSystem, The system to which this list belongs.
914 * pmEvent, Event from which we start drawing.
915 * pmSelectedEvent, Event selected if any.
916 * Return values :
917 * NONE.
918 * History :
919 * K.Y., 18/06/99, Initial typing.
920 * Note :
921 ******************************************************************/
fbbb2697 922static void WDFillEventList(GtkWidget* pmList,
fcdf0ec2 923 db* pmTraceDB,
924 systemInfo* pmSystem,
925 event* pmEvent,
926 event* pmSelectedEvent)
927{
928 gint i = 0; /* Generic index */
929 event lEvent; /* Generic event */
930 gchar lTimeStr[TIME_STR_LEN]; /* Time of event */
931 static gchar* lString[RTCLIST_NB_COLUMNS]={'\0'}; /* Strings describing event */
932 process* pProcess; /* Generic process pointer */
933#if SUPP_RTAI
934 RTAItask* pTask = NULL; /* Generic task pointer */
935#endif /* SUPP_RTAI */
936 eventDescription lEventDesc; /* Description of event */
937
938 /* Did we allocate space for strings */
939 if(lString[0] == NULL)
940 /* Allocate space for strings */
941 for (i = 0; i < RTCLIST_NB_COLUMNS - 1; i++)
942 lString[i] = (char*) g_malloc(MW_DEFAULT_STRLEN);
943
944 /* Allocate space for description string */
945 lString[RTCLIST_NB_COLUMNS - 1] = (char*) g_malloc(MW_LONG_STRLEN);
946
947 /* If no event was supplied, start at the beginning */
948 if(pmEvent == NULL)
949 lEvent = pmTraceDB->FirstEvent;
950 else
951 lEvent = *pmEvent;
952
953 /* Freeze and clear clist */
954 gtk_clist_freeze(GTK_CLIST(pmList));
955 gtk_clist_clear(GTK_CLIST(pmList));
956
957 /* Reset index */
958 i = 0;
959
960 /* Go through the event list */
961 do
962 {
963 /* Get the event description */
964 DBEventDescription(pmTraceDB, &lEvent, TRUE, &lEventDesc);
965
966 /* Get the event's process */
967 pProcess = DBEventProcess(pmTraceDB, &lEvent, pmSystem, FALSE);
968
969#if SUPP_RTAI
970 /* Does this trace contain RTAI information */
971 if(pmTraceDB->SystemType == TRACE_SYS_TYPE_RTAI_LINUX)
972 /* Get the RTAI task to which this event belongs */
973 pTask = RTAIDBEventTask(pmTraceDB, &lEvent, pmSystem, FALSE);
974#endif /* SUPP_RTAI */
975
976 /* Set the event's entry in the list of raw events displayed */
977 sRawEventsDisplayed[i] = lEvent;
978
979 /* Add text describing the event */
980 /* The CPU ID */
981 if(pmTraceDB->LogCPUID == TRUE)
982 snprintf(lString[0], MW_DEFAULT_STRLEN, "%d", lEventDesc.CPUID);
983 else
984 snprintf(lString[0], MW_DEFAULT_STRLEN, "0");
985
986 /* The event ID */
987 snprintf(lString[1], MW_DEFAULT_STRLEN, "%s", pmTraceDB->EventString(pmTraceDB, lEventDesc.ID, &lEvent));
988
989 /* The event's time of occurence */
990 DBFormatTimeInReadableString(lTimeStr,
991 lEventDesc.Time.tv_sec,
992 lEventDesc.Time.tv_usec);
993 snprintf(lString[2], MW_DEFAULT_STRLEN, "%s", lTimeStr);
994
995 /* Is this an RT event */
996 if(lEventDesc.ID <= TRACE_MAX)
997 {
998 /* The PID of the process to which the event belongs */
999 if(pProcess != NULL)
1000 snprintf(lString[3], MW_DEFAULT_STRLEN, "%d", pProcess->PID);
1001 else
1002 snprintf(lString[3], MW_DEFAULT_STRLEN, "N/A");
1003 }
1004#if SUPP_RTAI
1005 else
1006 {
1007 /* The TID of the task to which the event belongs */
1008 if(pTask != NULL)
1009 snprintf(lString[3], MW_DEFAULT_STRLEN, "RT:%d", pTask->TID);
1010 else
1011 snprintf(lString[3], MW_DEFAULT_STRLEN, "RT:N/A");
1012 }
1013#endif /* SUPP_RTAI */
1014
1015 /* The size of the entry */
1016 snprintf(lString[4], MW_DEFAULT_STRLEN, "%d", lEventDesc.Size);
1017
1018 /* The string describing the event */
1019 snprintf(lString[5], MW_LONG_STRLEN, "%s", lEventDesc.String);
1020
1021 /* Insert the entry into the list */
1022 gtk_clist_append(GTK_CLIST(pmList), lString);
1023
1024 /* Set the row's data to point to the current event */
1025 WDI_gtk_clist_set_last_row_data_full(GTK_CLIST(pmList), (gpointer) &(sRawEventsDisplayed[i]), NULL);
1026
1027 /* Was this the last selected event */
1028 if(DBEventsEqual(lEvent, (*pmSelectedEvent)))
1029 gtk_clist_select_row(GTK_CLIST(pmList), i, 0);
1030
1031 /* Go to next row */
1032 i++;
1033 } while((DBEventNext(pmTraceDB, &lEvent) == TRUE) && (i < RTCLIST_NB_ROWS));
1034
1035 /* Resize the list's length */
1036 gtk_widget_queue_resize(pmList);
1037
1038 /* Thaw the clist */
1039 gtk_clist_thaw(GTK_CLIST(pmList));
1040}
1041
fbbb2697 1042#endif //DEBUG
1043
1044static void destroy_cb( GtkWidget *widget,
1045 gpointer data )
1046{
1047 gtk_main_quit ();
1048}
1049
1050
1051
1052int main(int argc, char **argv)
1053{
1054 GtkWidget *Window;
1055 GtkWidget *ListViewer;
1056 GtkWidget *VBox_V;
1057 EventViewerData *Event_Viewer_Data;
1058
1059 /* Initialize i18n support */
1060 gtk_set_locale ();
1061
1062 /* Initialize the widget set */
1063 gtk_init (&argc, &argv);
1064
1065 init();
fcdf0ec2 1066
fbbb2697 1067 Window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
1068 gtk_window_set_title (GTK_WINDOW (Window), ("Test Window"));
1069
1070 g_signal_connect (G_OBJECT (Window), "destroy",
1071 G_CALLBACK (destroy_cb), NULL);
1072
1073
1074 VBox_V = gtk_vbox_new(0, 0);
1075 gtk_container_add (GTK_CONTAINER (Window), VBox_V);
1076
1077 ListViewer = hGuiEvents(Window);
1078 gtk_box_pack_start(GTK_BOX(VBox_V), ListViewer, TRUE, TRUE, 0);
1079
6601bf27 1080 //ListViewer = hGuiEvents(Window);
1081 //gtk_box_pack_start(GTK_BOX(VBox_V), ListViewer, FALSE, TRUE, 0);
fbbb2697 1082
1083 gtk_widget_show (VBox_V);
1084 gtk_widget_show (Window);
1085
1086 gtk_main ();
1087
1088 g_critical("main loop finished");
1089
1090 //hGuiEvents_Destructor(ListViewer);
1091
1092 //g_critical("GuiEvents Destructor finished");
1093 destroy();
1094
1095 return 0;
1096}
fcdf0ec2 1097
17abcce3 1098
5c7463ed 1099/*\@}*/
fbbb2697 1100
This page took 0.067929 seconds and 4 git commands to generate.