/* Chrysalide - Outil d'analyse de fichiers binaires * breaks.c - panneau d'affichage des points d'arrêt * * Copyright (C) 2010-2017 Cyrille Bagard * * This file is part of Chrysalide. * * Chrysalide is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * Chrysalide is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "breaks.h" #include "panel-int.h" #include "../debug/break.h" #include "../gtkext/easygtk.h" #include "../gtkext/support.h" #define _(str) str /* Panneau d'affichage des points d'arrêt (instance) */ struct _GBreaksPanel { GEditorPanel parent; /* A laisser en premier */ GtkTreeView *treeview; /* Composant d'affichage */ GtkTreeStore *store; /* Modèle de gestion */ GLoadedBinary *binary; /* Binaire en cours d'étude */ }; /* Panneau d'affichage des points d'arrêt (classe) */ struct _GBreaksPanelClass { GEditorPanelClass parent; /* A laisser en premier */ }; /* Colonnes de la liste des symboles */ typedef enum _BreaksColumn { BKC_POINT, /* Point d'arrêt de référence */ BKC_ICON, /* Statut du point d'arrêt */ BKC_ADDRESS, /* Adresse mémoire du symbole */ BKC_STRING, /* Désignation humaine */ BKC_COUNT /* Nombre de colonnes */ } BreaksColumn; /* Initialise la classe des panneaux de points d'arrêt. */ static void g_breaks_panel_class_init(GBreaksPanelClass *); /* Initialise une instance de panneau de points d'arrêt. */ static void g_breaks_panel_init(GBreaksPanel *); /* Réagit à un changement du binaire courant. */ static void reload_breaks_for_new_binary(GBreaksPanel *, GLoadedBinary *); /* Intègre à l'affichage un groupe de points d'arrêt. */ static void add_bp_group_to_breaks_panel(GLoadedBinary *, GBreakGroup *, GBreaksPanel *); /* Réagit à une nouvelle création de point d'arrêt. */ static void refresh_breaks_panel_on_bp_added(GBreakGroup *, GBreakPoint *, GBreaksPanel *); /* Réagit à une suppression de point d'arrêt. */ static void refresh_breaks_panel_on_bp_removed(GBreakGroup *, GBreakPoint *, GBreaksPanel *); /* Réagit à une modification de point d'arrêt. */ static void refresh_breaks_panel_on_bp_changed(GBreakGroup *, GBreakPoint *, GBreaksPanel *); /* Retrouve un point d'arrêt dans la liste affichée. */ static bool find_breakpoint_in_breaks_panel(GBreaksPanel *, GBreakPoint *, GtkTreeIter *); /* Indique le type défini pour un panneau d'affichage des points d'arrêt. */ G_DEFINE_TYPE(GBreaksPanel, g_breaks_panel, G_TYPE_EDITOR_PANEL); /****************************************************************************** * * * Paramètres : klass = classe à initialiser. * * * * Description : Initialise la classe des panneaux de points d'arrêt. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void g_breaks_panel_class_init(GBreaksPanelClass *klass) { } /****************************************************************************** * * * Paramètres : panel = instance à initialiser. * * * * Description : Initialise une instance de panneau de points d'arrêt. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void g_breaks_panel_init(GBreaksPanel *panel) { GEditorPanel *base; /* Version basique d'instance */ GObject *ref; /* Espace de référencement */ GtkWidget *toolbar; /* Barre d'outils */ GtkWidget *button; /* Bouton de cette même barre */ GtkWidget *separator; /* Barre de séparation vert. */ GtkWidget *scrollwnd; /* Support défilant */ GtkWidget *treeview; /* Affichage de la liste */ GtkCellRenderer *renderer; /* Moteur de rendu de colonne */ GtkTreeViewColumn *column; /* Colonne de la liste */ GtkTreeSelection *select; /* Sélection dans la liste */ base = G_EDITOR_PANEL(panel); base->name = _("Breakpoints"); base->reload_binary = (reload_for_new_binary_fc)reload_breaks_for_new_binary; base->widget = gtk_box_new(GTK_ORIENTATION_VERTICAL, 8); gtk_widget_show(base->widget); ref = G_OBJECT(base->widget); g_object_set_data(ref, "panel", panel); scrollwnd = gtk_scrolled_window_new(NULL, NULL); gtk_widget_show(scrollwnd); gtk_box_pack_start(GTK_BOX(base->widget), scrollwnd, TRUE, TRUE, 0); gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrollwnd), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrollwnd), GTK_SHADOW_IN); panel->store = gtk_tree_store_new(BKC_COUNT, G_TYPE_OBJECT, GDK_TYPE_PIXBUF, G_TYPE_STRING, G_TYPE_STRING); treeview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(panel->store)); gtk_widget_show(treeview); gtk_container_add(GTK_CONTAINER(scrollwnd), treeview); panel->treeview = GTK_TREE_VIEW(treeview); g_object_unref(G_OBJECT(panel->store)); column = gtk_tree_view_column_new(); gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column); gtk_tree_view_set_expander_column(GTK_TREE_VIEW(treeview), column); column = gtk_tree_view_column_new(); gtk_tree_view_column_set_title(column, _("Address")); renderer = gtk_cell_renderer_pixbuf_new(); gtk_tree_view_column_pack_start(column, renderer, FALSE); gtk_tree_view_column_add_attribute(column, renderer, "pixbuf", BKC_ICON); renderer = gtk_cell_renderer_text_new(); gtk_tree_view_column_pack_start(column, renderer, TRUE); gtk_tree_view_column_add_attribute(column, renderer, "text", BKC_ADDRESS); gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column); //renderer = gtk_cell_renderer_text_new(); //column = gtk_tree_view_column_new_with_attributes(_("Address"), renderer, "text", BKC_ADDRESS, NULL); //gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column); renderer = gtk_cell_renderer_text_new(); column = gtk_tree_view_column_new_with_attributes(_("String"), renderer, "text", BKC_STRING, NULL); gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column); } /****************************************************************************** * * * Paramètres : - * * * * Description : Crée un panneau d'aperçu de graphiques. * * * * Retour : Adresse de la structure mise en place. * * * * Remarques : - * * * ******************************************************************************/ GEditorPanel *g_breaks_panel_new(void) { GEditorPanel *result; /* Structure à retourner */ result = g_object_new(G_TYPE_BREAKS_PANEL, NULL); return G_EDITOR_PANEL(result); } /****************************************************************************** * * * Paramètres : panel = panneau à mettre à jour. * * binary = nouvelle instance de binaire analysé. * * * * Description : Réagit à un changement du binaire courant. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void reload_breaks_for_new_binary(GBreaksPanel *panel, GLoadedBinary *binary) { printf("CHNAGE BINARY !\n"); panel->binary = binary; //g_loaded_binary_for_each_bp_group(binary, (GExtFunc)add_bp_group_to_breaks_panel, panel); } /****************************************************************************** * * * Paramètres : binary = instance active du binaire analysé. * * group = groupe de points d'arrêt présent dans le binaire. * * panel = panneau à mettre à jour. * * * * Description : Intègre à l'affichage un groupe de points d'arrêt. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void add_bp_group_to_breaks_panel(GLoadedBinary *binary, GBreakGroup *group, GBreaksPanel *panel) { /* FIXME : bloquer toute émission de signal tant que les ajouts ne sont pas terminés. */ g_signal_connect(group, "added", G_CALLBACK(refresh_breaks_panel_on_bp_added), panel); g_signal_connect(group, "removed", G_CALLBACK(refresh_breaks_panel_on_bp_removed), panel); g_break_group_for_each(group, (GExtFunc)refresh_breaks_panel_on_bp_added, panel); } /****************************************************************************** * * * Paramètres : group = ensemble de points d'arrêt intervenant. * * point = point d'arrêt à l'origine de la procédure. * * panel = panneau à mettre à jour. * * * * Description : Réagit à une nouvelle création de point d'arrêt. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void refresh_breaks_panel_on_bp_added(GBreakGroup *group, GBreakPoint *point, GBreaksPanel *panel) { #if 0 GExeFormat *format; /* Format associé au binaire */ GArchProcessor *proc; /* Architecture utilisée */ char address[VMPA_MAX_SIZE]; /* Conversion de l'adresse */ GtkTreeIter iter; /* Point d'insertion */ format = g_loaded_binary_get_format(panel->binary); proc = get_arch_processor_from_format(format); vmpa_to_string(g_break_point_get_address(point), g_arch_processor_get_memory_size(proc), address); gtk_tree_store_append(panel->store, &iter, NULL); gtk_tree_store_set(panel->store, &iter, BKC_POINT, point, BKC_ADDRESS, address, BKC_STRING, "???", -1); /* Pour le reste... */ refresh_breaks_panel_on_bp_changed(group, point, panel); g_object_unref(G_OBJECT(format)); #endif } /****************************************************************************** * * * Paramètres : group = ensemble de points d'arrêt intervenant. * * point = point d'arrêt à l'origine de la procédure. * * panel = panneau à mettre à jour. * * * * Description : Réagit à une suppression de point d'arrêt. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void refresh_breaks_panel_on_bp_removed(GBreakGroup *group, GBreakPoint *point, GBreaksPanel *panel) { GtkTreeIter iter; /* Point de modification */ if (!find_breakpoint_in_breaks_panel(panel, point, &iter)) return; gtk_tree_store_remove(panel->store, &iter); } /****************************************************************************** * * * Paramètres : group = ensemble de points d'arrêt intervenant. * * point = point d'arrêt à l'origine de la procédure. * * panel = panneau à mettre à jour. * * * * Description : Réagit à une modification de point d'arrêt. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void refresh_breaks_panel_on_bp_changed(GBreakGroup *group, GBreakPoint *point, GBreaksPanel *panel) { GtkTreeIter iter; /* Point de modification */ GdkPixbuf *pixbuf; /* Tampon d'image chargé */ if (!find_breakpoint_in_breaks_panel(panel, point, &iter)) return; pixbuf = get_pixbuf_from_file("breakpoint_normal.png"); if (pixbuf == NULL) return; gtk_tree_store_set(panel->store, &iter, BKC_ICON, pixbuf, -1); } /****************************************************************************** * * * Paramètres : panel = panneau d'affichage à consulter. * * point = point d'arrêt lié à la procédure. * * * * Description : Retrouve un point d'arrêt dans la liste affichée. * * * * Retour : Bilan de la recherche. * * * * Remarques : - * * * ******************************************************************************/ static bool find_breakpoint_in_breaks_panel(GBreaksPanel *panel, GBreakPoint *point, GtkTreeIter *iter) { bool result; /* Bilan à retourner */ GtkTreeIter tmp; /* Sauvegarde temporaire */ bool test; /* Valide la poursuite */ GBreakPoint *bp; /* Adresse à comparer */ result = false; for (test = gtk_tree_model_get_iter_first(GTK_TREE_MODEL(panel->store), &tmp); test && !result; test = gtk_tree_model_iter_next(GTK_TREE_MODEL(panel->store), &tmp)) { gtk_tree_model_get(GTK_TREE_MODEL(panel->store), &tmp, BKC_POINT, &bp, -1); if (bp == point) { *iter = tmp; result = true; } } return result; }