+char * get_unload_module(char ** loaded_module_name, int nb_module)
+{
+ GtkWidget * dialogue;
+ GtkWidget * scroll_win;
+ GtkWidget * tree;
+ GtkListStore * store;
+ GtkTreeViewColumn * column;
+ GtkCellRenderer * renderer;
+ GtkTreeSelection * select;
+ GtkTreeIter iter;
+ gint id, i;
+ char * unload_module_name = NULL;
+
+ dialogue = gtk_dialog_new_with_buttons("Select an unload module",
+ NULL,
+ GTK_DIALOG_MODAL,
+ GTK_STOCK_OK,GTK_RESPONSE_ACCEPT,
+ GTK_STOCK_CANCEL,GTK_RESPONSE_REJECT,
+ NULL);
+ gtk_window_set_default_size((GtkWindow*)dialogue, 500, 200);
+
+ scroll_win = gtk_scrolled_window_new (NULL, NULL);
+ gtk_widget_show ( scroll_win);
+ gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scroll_win),
+ GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
+
+ store = gtk_list_store_new (N_COLUMNS,G_TYPE_STRING);
+ tree = gtk_tree_view_new_with_model(GTK_TREE_MODEL (store));
+ gtk_widget_show ( tree);
+ g_object_unref (G_OBJECT (store));
+
+ renderer = gtk_cell_renderer_text_new ();
+ column = gtk_tree_view_column_new_with_attributes ("MODULE NAME",
+ renderer,
+ "text", MODULE_COLUMN,
+ NULL);
+ gtk_tree_view_column_set_alignment (column, 0.5);
+ gtk_tree_view_column_set_fixed_width (column, 150);
+ gtk_tree_view_append_column (GTK_TREE_VIEW (tree), column);
+
+ select = gtk_tree_view_get_selection (GTK_TREE_VIEW (tree));
+ gtk_tree_selection_set_mode (select, GTK_SELECTION_SINGLE);
+
+ gtk_container_add (GTK_CONTAINER (scroll_win), tree);
+
+ gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialogue)->vbox), scroll_win,TRUE, TRUE,0);
+
+ for(i=0;i<nb_module;i++){
+ gtk_list_store_append (store, &iter);
+ gtk_list_store_set (store, &iter, MODULE_COLUMN,loaded_module_name[i],-1);
+ }
+
+ id = gtk_dialog_run(GTK_DIALOG(dialogue));
+ switch(id){
+ case GTK_RESPONSE_ACCEPT:
+ case GTK_RESPONSE_OK:
+ if (gtk_tree_selection_get_selected (select, (GtkTreeModel**)&store, &iter)){
+ gtk_tree_model_get ((GtkTreeModel*)store, &iter, MODULE_COLUMN, &unload_module_name, -1);
+ }
+ case GTK_RESPONSE_REJECT:
+ case GTK_RESPONSE_CANCEL:
+ default:
+ gtk_widget_destroy(dialogue);
+ break;
+ }
+
+ return unload_module_name;
+}
+
+void destroy_hash_key(gpointer key)
+{
+ g_free(key);
+}
+
+void destroy_hash_data(gpointer data)
+{
+}
+
+
+void insertMenuToolbarItem(mainWindow * mw)
+{
+ int i;
+ GdkPixbuf *pixbuf;
+ view_constructor constructor;
+ LttvMenus * menu;
+ LttvToolbars * toolbar;
+ lttv_menu_closure *menuItem;
+ lttv_toolbar_closure *toolbarItem;
+ LttvAttributeValue value;
+ LttvIAttribute *attributes = LTTV_IATTRIBUTE(lttv_global_attributes());
+ GtkWidget * ToolMenuTitle_menu, *insert_view, *pixmap;
+
+ g_assert(lttv_iattribute_find_by_path(attributes,
+ "viewers/menu", LTTV_POINTER, &value));
+ menu = (LttvMenus*)*(value.v_pointer);
+
+ if(menu){
+ for(i=0;i<menu->len;i++){
+ menuItem = &g_array_index(menu, lttv_menu_closure, i);
+ constructor = menuItem->con;
+ ToolMenuTitle_menu = lookup_widget(mw->MWindow,"ToolMenuTitle_menu");
+ insert_view = gtk_menu_item_new_with_mnemonic (menuItem->menuText);
+ gtk_widget_show (insert_view);
+ gtk_container_add (GTK_CONTAINER (ToolMenuTitle_menu), insert_view);
+ g_signal_connect ((gpointer) insert_view, "activate",
+ G_CALLBACK (insertViewTest),
+ constructor);
+ g_hash_table_insert(mw->hash_menu_item, g_strdup(menuItem->menuText),
+ insert_view);
+ }
+ }
+
+ g_assert(lttv_iattribute_find_by_path(attributes,
+ "viewers/toolbar", LTTV_POINTER, &value));
+ toolbar = (LttvToolbars*)*(value.v_pointer);
+
+ if(toolbar){
+ for(i=0;i<toolbar->len;i++){
+ toolbarItem = &g_array_index(toolbar, lttv_toolbar_closure, i);
+ constructor = toolbarItem->con;
+ ToolMenuTitle_menu = lookup_widget(mw->MWindow,"MToolbar2");
+ pixbuf = gdk_pixbuf_new_from_xpm_data ((const char**)toolbarItem->pixmap);
+ pixmap = gtk_image_new_from_pixbuf(pixbuf);
+ insert_view = gtk_toolbar_append_element (GTK_TOOLBAR (ToolMenuTitle_menu),
+ GTK_TOOLBAR_CHILD_BUTTON,
+ NULL,
+ "",
+ toolbarItem->tooltip, NULL,
+ pixmap, NULL, NULL);
+ gtk_label_set_use_underline (GTK_LABEL (((GtkToolbarChild*) (g_list_last (GTK_TOOLBAR (ToolMenuTitle_menu)->children)->data))->label), TRUE);
+ gtk_widget_show (insert_view);
+ gtk_container_set_border_width (GTK_CONTAINER (insert_view), 1);
+ g_signal_connect ((gpointer) insert_view, "clicked",G_CALLBACK (insertViewTest),constructor);
+ g_hash_table_insert(mw->hash_toolbar_item, g_strdup(toolbarItem->tooltip),
+ insert_view);
+ }
+ }
+}
+
+void constructMainWin(mainWindow * parent, WindowCreationData * win_creation_data,
+ gboolean first_window)
+{
+ g_critical("constructMainWin()");
+ GtkWidget * newWindow; /* New generated main window */
+ mainWindow * newMWindow;/* New main window structure */
+ GtkNotebook * notebook;
+ LttvIAttribute *attributes =
+ LTTV_IATTRIBUTE(g_object_new(LTTV_ATTRIBUTE_TYPE, NULL));
+ LttvAttributeValue value;
+
+ newMWindow = g_new(mainWindow, 1);
+
+ // Add the object's information to the module's array
+ Main_Window_List = g_slist_append(Main_Window_List, newMWindow);
+
+
+ newWindow = create_MWindow();
+ gtk_widget_show (newWindow);
+
+ newMWindow->Attributes = attributes;
+
+ newMWindow->Traceset_Info = g_new(TracesetInfo,1);
+ newMWindow->Traceset_Info->path = NULL ;
+
+
+ newMWindow->Traceset_Info->before_traceset = lttv_hooks_new();
+ newMWindow->Traceset_Info->after_traceset = lttv_hooks_new();
+ newMWindow->Traceset_Info->before_trace = lttv_hooks_new();
+ newMWindow->Traceset_Info->after_trace = lttv_hooks_new();
+ newMWindow->Traceset_Info->before_tracefile = lttv_hooks_new();
+ newMWindow->Traceset_Info->after_tracefile = lttv_hooks_new();
+ newMWindow->Traceset_Info->before_event = lttv_hooks_new();
+ newMWindow->Traceset_Info->after_event = lttv_hooks_new();
+
+ g_assert(lttv_iattribute_find_by_path(attributes, "hooks/traceset/before",
+ LTTV_POINTER, &value));
+ *(value.v_pointer) = newMWindow->Traceset_Info->before_traceset;
+ g_assert(lttv_iattribute_find_by_path(attributes, "hooks/traceset/after",
+ LTTV_POINTER, &value));
+ *(value.v_pointer) = newMWindow->Traceset_Info->after_traceset;
+ g_assert(lttv_iattribute_find_by_path(attributes, "hooks/trace/before",
+ LTTV_POINTER, &value));
+ *(value.v_pointer) = newMWindow->Traceset_Info->before_trace;
+ g_assert(lttv_iattribute_find_by_path(attributes, "hooks/trace/after",
+ LTTV_POINTER, &value));
+ *(value.v_pointer) = newMWindow->Traceset_Info->after_trace;
+ g_assert(lttv_iattribute_find_by_path(attributes, "hooks/tracefile/before",
+ LTTV_POINTER, &value));
+ *(value.v_pointer) = newMWindow->Traceset_Info->before_tracefile;
+ g_assert(lttv_iattribute_find_by_path(attributes, "hooks/tracefile/after",
+ LTTV_POINTER, &value));
+ *(value.v_pointer) = newMWindow->Traceset_Info->after_tracefile;
+ g_assert(lttv_iattribute_find_by_path(attributes, "hooks/event/before",
+ LTTV_POINTER, &value));
+ *(value.v_pointer) = newMWindow->Traceset_Info->before_event;
+ g_assert(lttv_iattribute_find_by_path(attributes, "hooks/event/after",
+ LTTV_POINTER, &value));
+ *(value.v_pointer) = newMWindow->Traceset_Info->after_event;
+
+
+ newMWindow->MWindow = newWindow;
+ newMWindow->Tab = NULL;
+ newMWindow->CurrentTab = NULL;
+ newMWindow->Attributes = LTTV_IATTRIBUTE(g_object_new(LTTV_ATTRIBUTE_TYPE, NULL));
+ if(parent){
+ newMWindow->Traceset_Info->traceset =
+ lttv_traceset_copy(parent->Traceset_Info->traceset);
+
+//FIXME copy not implemented in lower level
+ newMWindow->Traceset_Info->TracesetContext =
+ g_object_new(LTTV_TRACESET_STATS_TYPE, NULL);
+ lttv_context_init(
+ LTTV_TRACESET_CONTEXT(newMWindow->Traceset_Info->TracesetContext),
+ newMWindow->Traceset_Info->traceset);
+ //newMWindow->traceset_context = parent->traceset_context;
+ newMWindow->winCreationData = parent->winCreationData;
+ }else{
+ newMWindow->Traceset_Info->traceset = lttv_traceset_new();
+
+ /* Add the command line trace */
+ if(gInit_Trace != NULL && first_window)
+ lttv_traceset_add(newMWindow->Traceset_Info->traceset, gInit_Trace);
+ /* NOTE : the context must be recreated if we change the traceset,
+ * ie : adding/removing traces */
+ newMWindow->Traceset_Info->TracesetContext =
+ g_object_new(LTTV_TRACESET_STATS_TYPE, NULL);
+ lttv_context_init(
+ LTTV_TRACESET_CONTEXT(newMWindow->Traceset_Info->TracesetContext),
+ newMWindow->Traceset_Info->traceset);
+
+ newMWindow->winCreationData = win_creation_data;
+ }
+
+ newMWindow->hash_menu_item = g_hash_table_new_full (g_str_hash, g_str_equal,
+ destroy_hash_key, destroy_hash_data);
+ newMWindow->hash_toolbar_item = g_hash_table_new_full (g_str_hash, g_str_equal,
+ destroy_hash_key, destroy_hash_data);
+
+ insertMenuToolbarItem(newMWindow);
+
+ g_object_set_data(G_OBJECT(newWindow), "mainWindow", (gpointer)newMWindow);
+
+ //create a default tab
+ notebook = (GtkNotebook *)lookup_widget(newMWindow->MWindow, "MNotebook");
+ if(notebook == NULL){
+ g_printf("Notebook does not exist\n");
+ return;
+ }
+ //for now there is no name field in LttvTraceset structure
+ //Use "Traceset" as the label for the default tab
+ create_tab(newMWindow->MWindow, notebook,"Traceset");
+
+ g_object_set_data_full(
+ G_OBJECT(newMWindow->MWindow),
+ "Main_Window_Data",
+ newMWindow,
+ (GDestroyNotify)mainWindow_free);
+
+ gWinCount++;
+}
+
+void Tab_Destructor(tab *Tab)
+{
+ if(Tab->Attributes)
+ g_object_unref(Tab->Attributes);
+
+ if(Tab->mw->Tab == Tab){
+ Tab->mw->Tab = Tab->Next;
+ }else{
+ tab * tmp1, *tmp = Tab->mw->Tab;
+ while(tmp != Tab){
+ tmp1 = tmp;
+ tmp = tmp->Next;
+ }
+ tmp1->Next = Tab->Next;
+ }
+ g_free(Tab);
+}
+
+void * create_tab(GtkWidget* parent, GtkNotebook * notebook, char * label)
+{
+ GList * list;
+ tab * tmpTab;
+ mainWindow * mwData;
+ LttTime TmpTime;
+
+ mwData = get_window_data_struct(parent);
+ tmpTab = mwData->Tab;
+ while(tmpTab && tmpTab->Next) tmpTab = tmpTab->Next;
+ if(!tmpTab){
+ mwData->CurrentTab = NULL;
+ tmpTab = g_new(tab,1);
+ mwData->Tab = tmpTab;
+ }else{
+ tmpTab->Next = g_new(tab,1);
+ tmpTab = tmpTab->Next;
+ }
+ if(mwData->CurrentTab){
+ // Will have to read directly at the main window level, as we want
+ // to be able to modify a traceset on the fly.
+ // tmpTab->traceStartTime = mwData->CurrentTab->traceStartTime;
+ // tmpTab->traceEndTime = mwData->CurrentTab->traceEndTime;
+ tmpTab->Time_Window = mwData->CurrentTab->Time_Window;
+ tmpTab->currentTime = mwData->CurrentTab->currentTime;
+ }else{
+ // Will have to read directly at the main window level, as we want
+ // to be able to modify a traceset on the fly.
+ // getTracesetTimeSpan(mwData,&tmpTab->traceStartTime, &tmpTab->traceEndTime);
+ tmpTab->Time_Window.startTime =
+ LTTV_TRACESET_CONTEXT(mwData->Traceset_Info->TracesetContext)->Time_Span->startTime;
+ if(DEFAULT_TIME_WIDTH_S <
+ LTTV_TRACESET_CONTEXT(mwData->Traceset_Info->TracesetContext)->Time_Span->endTime.tv_sec)
+ TmpTime.tv_sec = DEFAULT_TIME_WIDTH_S;
+ else
+ TmpTime.tv_sec =
+ LTTV_TRACESET_CONTEXT(mwData->Traceset_Info->TracesetContext)->Time_Span->endTime.tv_sec;
+ TmpTime.tv_nsec = 0;
+ tmpTab->Time_Window.Time_Width = TmpTime ;
+ tmpTab->currentTime.tv_sec = TmpTime.tv_sec / 2;
+ tmpTab->currentTime.tv_nsec = 0 ;
+ }
+ tmpTab->Attributes = LTTV_IATTRIBUTE(g_object_new(LTTV_ATTRIBUTE_TYPE, NULL));
+ // mwData->CurrentTab = tmpTab;
+ tmpTab->custom = (GtkCustom*)gtk_custom_new();
+ tmpTab->custom->mw = mwData;
+ gtk_widget_show((GtkWidget*)tmpTab->custom);
+ tmpTab->Next = NULL;
+ tmpTab->mw = mwData;
+
+ tmpTab->label = gtk_label_new (label);
+ gtk_widget_show (tmpTab->label);
+
+ g_object_set_data_full(
+ G_OBJECT(tmpTab->custom),
+ "Tab_Info",
+ tmpTab,
+ (GDestroyNotify)Tab_Destructor);
+
+ gtk_notebook_append_page(notebook, (GtkWidget*)tmpTab->custom, tmpTab->label);
+ list = gtk_container_get_children(GTK_CONTAINER(notebook));
+ gtk_notebook_set_current_page(notebook,g_list_length(list)-1);
+}
+
+void remove_menu_item(gpointer main_win, gpointer user_data)
+{
+ mainWindow * mw = (mainWindow *) main_win;
+ lttv_menu_closure *menuItem = (lttv_menu_closure *)user_data;
+ GtkWidget * ToolMenuTitle_menu, *insert_view;
+
+ ToolMenuTitle_menu = lookup_widget(mw->MWindow,"ToolMenuTitle_menu");
+ insert_view = (GtkWidget*)g_hash_table_lookup(mw->hash_menu_item,
+ menuItem->menuText);
+ if(insert_view){
+ g_hash_table_remove(mw->hash_menu_item, menuItem->menuText);
+ gtk_container_remove (GTK_CONTAINER (ToolMenuTitle_menu), insert_view);
+ }
+}
+
+void remove_toolbar_item(gpointer main_win, gpointer user_data)
+{
+ mainWindow * mw = (mainWindow *) main_win;
+ lttv_toolbar_closure *toolbarItem = (lttv_toolbar_closure *)user_data;
+ GtkWidget * ToolMenuTitle_menu, *insert_view;
+
+
+ ToolMenuTitle_menu = lookup_widget(mw->MWindow,"MToolbar2");
+ insert_view = (GtkWidget*)g_hash_table_lookup(mw->hash_toolbar_item,
+ toolbarItem->tooltip);
+ if(insert_view){
+ g_hash_table_remove(mw->hash_toolbar_item, toolbarItem->tooltip);
+ gtk_container_remove (GTK_CONTAINER (ToolMenuTitle_menu), insert_view);
+ }
+}
+
+/**
+ * Remove menu and toolbar item when a module unloaded
+ */
+void main_window_remove_menu_item(lttv_constructor constructor)
+{
+ int i;
+ LttvMenus * menu;
+ lttv_menu_closure *menuItem;
+ LttvAttributeValue value;
+ LttvIAttribute *attributes = LTTV_IATTRIBUTE(lttv_global_attributes());
+
+ g_assert(lttv_iattribute_find_by_path(attributes,
+ "viewers/menu", LTTV_POINTER, &value));
+ menu = (LttvMenus*)*(value.v_pointer);
+
+ if(menu){
+ for(i=0;i<menu->len;i++){
+ menuItem = &g_array_index(menu, lttv_menu_closure, i);
+ if(menuItem->con != constructor) continue;
+ g_slist_foreach(Main_Window_List, remove_menu_item, menuItem);
+ break;
+ }
+ }
+
+}
+
+void main_window_remove_toolbar_item(lttv_constructor constructor)
+{
+ int i;
+ LttvToolbars * toolbar;
+ lttv_toolbar_closure *toolbarItem;
+ LttvAttributeValue value;
+ LttvIAttribute *attributes = LTTV_IATTRIBUTE(lttv_global_attributes());
+
+ g_assert(lttv_iattribute_find_by_path(attributes,
+ "viewers/toolbar", LTTV_POINTER, &value));
+ toolbar = (LttvToolbars*)*(value.v_pointer);
+
+ if(toolbar){
+ for(i=0;i<toolbar->len;i++){
+ toolbarItem = &g_array_index(toolbar, lttv_toolbar_closure, i);
+ if(toolbarItem->con != constructor) continue;
+ g_slist_foreach(Main_Window_List, remove_toolbar_item, toolbarItem);
+ break;
+ }
+ }
+}