From 783e5e1977c1e4dadf938befa9fce9a311079413 Mon Sep 17 00:00:00 2001 From: Cyrille Bagard Date: Wed, 18 Nov 2009 23:20:16 +0000 Subject: Saved the current work on binary parts selection. git-svn-id: svn://svn.gna.org/svn/chrysalide/trunk@137 abbe820e-26c8-41b2-8c08-b7b2b41f8b0a --- ChangeLog | 37 +++ configure.ac | 1 + src/Makefile.am | 6 +- src/analysis/binary.c | 87 ++++++- src/analysis/binary.h | 18 ++ src/dialogs/binparts.c | 604 +++++++++++++++++++++++++++++++++++++++++++++++ src/dialogs/binparts.h | 42 ++++ src/dlg_sections.c | 218 ----------------- src/dlg_sections.h | 42 ---- src/editor.c | 69 +++--- src/format/elf/elf.c | 15 +- src/format/elf/section.c | 2 + src/format/part.c | 19 ++ src/format/part.h | 3 + src/project.c | 4 +- 15 files changed, 872 insertions(+), 295 deletions(-) create mode 100644 src/dialogs/binparts.c create mode 100644 src/dialogs/binparts.h diff --git a/ChangeLog b/ChangeLog index c869692..6d4fcd5 100644 --- a/ChangeLog +++ b/ChangeLog @@ -1,3 +1,40 @@ +09-11-19 Cyrille Bagard + + * configure.ac: + Add the new Makefile from the 'src/dialogs/Makefile' directory to + AC_CONFIG_FILES. + + * src/analysis/binary.c: + * src/analysis/binary.h: + Handle several kinds of binary parts. + + * src/dialogs/binparts.c: + * src/dialogs/binparts.h: + New entries: create a dialog window to select binary parts to disassemble. + + * src/dlg_sections.c: + * src/dlg_sections.h: + Moved entries: see the src/dialogs/binparts.[ch] files. + + * src/editor.c: + Update code and add menus. + + * src/format/elf/elf.c: + Load the name of sections. + + * src/format/elf/section.c: + Typo: add a 'FIXME' comment. + + * src/format/part.c: + * src/format/part.h: + Provide a way to retrieve the name of a binary part. + + * src/Makefile.am: + Add dialogs/libdialogs.a to openida_LDADD. + + * src/project.c: + Fix a bug when saving projects. + 09-11-07 Cyrille Bagard * configure.ac: diff --git a/configure.ac b/configure.ac index d340b3c..11a178d 100644 --- a/configure.ac +++ b/configure.ac @@ -246,6 +246,7 @@ AC_CONFIG_FILES([Makefile src/debug/Makefile src/debug/ptrace/Makefile src/debug/remgdb/Makefile + src/dialogs/Makefile src/format/Makefile src/format/dwarf/Makefile src/format/elf/Makefile diff --git a/src/Makefile.am b/src/Makefile.am index 7e819de..e848c2b 100644 --- a/src/Makefile.am +++ b/src/Makefile.am @@ -73,7 +73,6 @@ liboidaplugin_la_LIBADD = \ openida_SOURCES = \ configuration.h configuration.c \ - dlg_sections.h dlg_sections.c \ editor.h editor.c \ main.c \ params.h params.c \ @@ -99,7 +98,8 @@ openida_LDFLAGS = $(LIBGTK_LIBS) -L/usr/X11R6/lib -ldl -lm $(LIBXML_LIBS) `pkg-c openida_LDADD = $(LIBINTL) \ debug/libdebug.a \ debug/remgdb/libdebugremgdb.a \ - debug/ptrace/libdebugptrace.a + debug/ptrace/libdebugptrace.a \ + dialogs/libdialogs.a @@ -109,4 +109,4 @@ openida_LDADD = $(LIBINTL) \ # gtkext doit être traité en premier, à cause des marshals GLib -SUBDIRS = glibext gtkext analysis arch format common debug graph panels plugins +SUBDIRS = glibext gtkext analysis arch format common debug dialogs graph panels plugins diff --git a/src/analysis/binary.c b/src/analysis/binary.c index 97b916e..d2c7f3e 100644 --- a/src/analysis/binary.c +++ b/src/analysis/binary.c @@ -136,6 +136,10 @@ struct _GOpenidaBinary GExeFormat *format; /* Format du binaire */ GArchProcessor *proc; /* Architecture du binaire */ + BinaryPartModel model; /* Modèle de sélection */ + GBinPart **parts[BPM_COUNT]; /* Parties binaires à analyser */ + size_t parts_count[BPM_COUNT]; /* Quantité de ces parties */ + GRenderingLine *lines; /* Lignes de rendu en place */ GRenderingOptions *options; /* Options de désassemblage */ @@ -549,6 +553,9 @@ static void limit_all_routines(GRenderingLine *lines, GBinRoutine **routines, si start = g_binary_routine_get_address(routines[i]); line = g_rendering_line_find_by_address(lines, NULL, start); + /* Si le symbole est hors du code analysé (routine de PLT par exemple) */ + if (line == NULL) continue; + last = find_best_ending_address_for_routine(line, i, starts, lengths, count); line = g_rendering_line_find_by_address(lines, NULL, last); @@ -821,6 +828,17 @@ bool g_openida_binary_save(const GOpenidaBinary *binary, xmlDocPtr xdoc, xmlXPat free(access); + + + access = strdup(path); + access = stradd(access, "/Filename2"); + + result &= add_content_to_node(xdoc, context, access, binary->filename); + + free(access); + + + return result; } @@ -828,6 +846,55 @@ bool g_openida_binary_save(const GOpenidaBinary *binary, xmlDocPtr xdoc, xmlXPat /****************************************************************************** * * +* Paramètres : binary = élément binaire à consulter. * +* parts = liste des zones binaires à analyser. * +* model = modèle de sélection des zones. * +* count = quantité de zones listées. * +* * +* Description : Définit les parties de binaire à analyser. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +void g_openida_binary_set_parts(GOpenidaBinary *binary, BinaryPartModel model, GBinPart **parts, size_t count) +{ + qsort(parts, count, sizeof(GBinPart *), g_binary_part_compare); + + binary->parts[model] = parts; + binary->parts_count[model] = count; + +} + + +/****************************************************************************** +* * +* Paramètres : binary = élément binaire à consulter. * +* model = modèle de sélection des zones. [OUT] * +* count = quantité de zones listées. [OUT] * +* * +* Description : Fournit les parties de binaire analysées. * +* * +* Retour : Zones binaires à analyser. * +* * +* Remarques : - * +* * +******************************************************************************/ + +GBinPart **g_openida_binary_get_parts(const GOpenidaBinary *binary, BinaryPartModel *model, size_t *count) +{ + *model = binary->model; + *count = binary->parts_count; + + return binary->parts; + +} + + +/****************************************************************************** +* * * Paramètres : binary = élément binaire à traiter. * * * * Description : Lance l'analyse d'un élément binaire chargé. * @@ -847,8 +914,24 @@ void g_openida_binary_analyse(GOpenidaBinary *binary) queue = get_work_queue(); - parts = g_exe_format_get_parts(binary->format, &parts_count); - qsort(parts, parts_count, sizeof(GBinPart *), g_binary_part_compare); + if (binary->parts[binary->model] != NULL) + { + parts = binary->parts[binary->model]; + parts_count = binary->parts_count[binary->model]; + } + else + { + if (binary->parts[BPM_DEFAULT] != NULL) + { + parts = binary->parts[BPM_DEFAULT]; + parts_count = binary->parts_count[BPM_DEFAULT]; + } + else + { + parts = g_exe_format_get_parts(binary->format, &parts_count); + qsort(parts, parts_count, sizeof(GBinPart *), g_binary_part_compare); + } + } disass = g_delayed_disassembly_new(binary, parts, parts_count); diff --git a/src/analysis/binary.h b/src/analysis/binary.h index ac5af77..d195fef 100644 --- a/src/analysis/binary.h +++ b/src/analysis/binary.h @@ -50,6 +50,18 @@ typedef struct _GOpenidaBinary GOpenidaBinary; typedef struct _GOpenidaBinaryClass GOpenidaBinaryClass; +/* Modèle de sélection des parties */ +typedef enum _BinaryPartModel +{ + BPM_DEFAULT, /* Selon le modèle par défaut */ + BPM_ROUTINES, /* Sélection par les routines */ + BPM_USER, /* Définitions utilisateur */ + + BPM_COUNT + +} BinaryPartModel; + + /* Indique le type défini pour une description de fichier binaire. */ GType g_openida_binary_get_type(void); @@ -62,6 +74,12 @@ GOpenidaBinary *g_openida_binary_new_from_xml(xmlXPathContextPtr, const char *); /* Ecrit une sauvegarde du binaire dans un fichier XML. */ bool g_openida_binary_save(const GOpenidaBinary *, xmlDocPtr, xmlXPathContextPtr, const char *); +/* Définit les parties de binaire à analyser. */ +void g_openida_binary_set_parts(GOpenidaBinary *, BinaryPartModel, GBinPart **, size_t); + +/* Fournit les parties de binaire analysées. */ +GBinPart **g_openida_binary_get_parts(const GOpenidaBinary *, BinaryPartModel *, size_t *); + /* Lance l'analyse d'un élément binaire chargé. */ void g_openida_binary_analyse(GOpenidaBinary *); diff --git a/src/dialogs/binparts.c b/src/dialogs/binparts.c new file mode 100644 index 0000000..1135348 --- /dev/null +++ b/src/dialogs/binparts.c @@ -0,0 +1,604 @@ + +/* OpenIDA - Outil d'analyse de fichiers binaires + * dlg_sections.h - boîte de dialogue permettant une sélection des sections + * + * Copyright (C) 2008 Cyrille Bagard + * + * This file is part of OpenIDA. + * + * OpenIDA 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. + * + * OpenIDA 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 Foobar. If not, see . + */ + + +#include "binparts.h" + + +#include "../format/format.h" +#include "../gtkext/easygtk.h" + + + +#define _(str) str + + + +/* Colonnes de la liste des symboles */ +typedef enum _PartsColumn +{ + PTC_ACTIVE, /* Zone de code active ? */ + PTC_NAME, /* Désignation humaine */ + PTC_START, /* Adresse de départ */ + PTC_END, /* Adresse d'arrivée (exclue) */ + + PTC_COUNT /* Nombre de colonnes */ + +} PartsColumn; + + + +/* Mémoire d'un modèle */ +typedef struct _parts_model +{ + gboolean *selected; /* Sélection ou non de parties */ + size_t count; /* Qté. de prises en compte */ + +} parts_model; + + + + + +/* Charge les sections sélectionnées pour le projet courant. */ +void load_project_sections(void *, GObject *); + + + + + +/* Affiche les parties désassemblées par défaut. */ +static void load_default_parts(GObject *); + +/* Affiche les parties désassemblées selon les routines. */ +static void load_routines_parts(GObject *); + + + +/* Réagit à un changement de modèle. */ +static void on_model_change(GtkComboBox *, GObject *); + +/* Réagit à un changement de sélection de partie. */ +static void on_part_selection_toggle(GtkCellRendererToggle *, gchar *, GObject *); + + + +/****************************************************************************** +* * +* Paramètres : binary = informations sur le binaire actuellement ouvert. * +* * +* Description : Construit la fenêtre de sélection des sections. * +* * +* Retour : Adresse de la fenêtre mise en place. * +* * +* Remarques : - * +* * +******************************************************************************/ + +GtkWidget *create_sections_dialog(GOpenidaBinary *binary) +{ + GtkWidget *result; /* Fenêtre à renvoyer */ + GObject *ref; /* Espace de référencements */ + + + + + + GtkWidget *vbox1; + //GtkWidget *hbox1; + + + GtkWidget *label; /* Etiquette à afficher */ + + + GtkWidget *combobox; + + GtkWidget *alignment; /* Adaptation de disposition */ + GtkWidget *frame; /* Support avec encadrement */ + + GtkWidget *vbox2; + GtkWidget *hbox2; + GtkWidget *comboboxentry; + GtkWidget *hbox3; + GtkWidget *scrolledwindow1; + + GtkTreeStore *store; /* Modèle de gestion */ + GtkWidget *treeview; /* Affichage de la liste */ + GtkCellRenderer *renderer; /* Moteur de rendu de colonne */ + GtkTreeViewColumn *column; /* Colonne de la liste */ + + + GtkWidget *vbuttonbox1; + GtkWidget *hbuttonbox1; + + GtkWidget *button; /* Bouton de commande */ + + result = gtk_window_new(GTK_WINDOW_TOPLEVEL); + gtk_widget_set_size_request(result, 600, 400); + gtk_container_set_border_width(GTK_CONTAINER(result), 8); + gtk_window_set_title(GTK_WINDOW(result), _("Sections selection")); + gtk_window_set_position(GTK_WINDOW(result), GTK_WIN_POS_CENTER); + gtk_window_set_default_size(GTK_WINDOW(result), 600, 400); + gtk_window_set_type_hint(GTK_WINDOW(result), GDK_WINDOW_TYPE_HINT_DIALOG); + + ref= G_OBJECT(result); + g_object_set_data(ref, "binary", binary); + + vbox1 = gtk_vbox_new(FALSE, 8); + gtk_widget_show(vbox1); + gtk_container_add(GTK_CONTAINER(result), vbox1); + + /* + hbox1 = gtk_hbox_new(FALSE, 8); + gtk_widget_show(hbox1); + gtk_box_pack_start(GTK_BOX(vbox1), hbox1, FALSE, FALSE, 0); + + label = qck_create_label(NULL, NULL, _("Binary :")); + gtk_box_pack_start(GTK_BOX(hbox1), label, FALSE, FALSE, 0); + + combobox = gtk_combo_box_new_text(); + gtk_widget_show(combobox); + gtk_box_pack_start(GTK_BOX(hbox1), combobox, TRUE, TRUE, 0); + */ + + frame = qck_create_frame(_("Content to display"), &alignment, 4, 4, 12, 0); + gtk_box_pack_start(GTK_BOX(vbox1), frame, TRUE, TRUE, 0); + + + + + vbox2 = gtk_vbox_new(FALSE, 8); + gtk_widget_show(vbox2); + gtk_container_add(GTK_CONTAINER(alignment), vbox2); + + hbox2 = gtk_hbox_new(FALSE, 8); + gtk_widget_show(hbox2); + gtk_box_pack_start(GTK_BOX(vbox2), hbox2, FALSE, FALSE, 0); + + label = qck_create_label(NULL, NULL, _("Model :")); + gtk_box_pack_start(GTK_BOX(hbox2), label, FALSE, FALSE, 0); + + comboboxentry = qck_create_combobox(ref, "models", G_CALLBACK(NULL), NULL); + gtk_box_pack_start(GTK_BOX(hbox2), comboboxentry, TRUE, TRUE, 0); + + + button = qck_create_button_with_img(NULL, NULL, "gtk-add", G_CALLBACK(NULL), NULL); + gtk_box_pack_start(GTK_BOX(hbox2), button, FALSE, FALSE, 0); + + button = qck_create_button_with_img(NULL, NULL, "gtk-remove", G_CALLBACK(NULL), NULL); + gtk_box_pack_start(GTK_BOX(hbox2), button, FALSE, FALSE, 0); + + + + + hbox3 = gtk_hbox_new(FALSE, 8); + gtk_widget_show(hbox3); + gtk_box_pack_start(GTK_BOX(vbox2), hbox3, TRUE, TRUE, 0); + + scrolledwindow1 = gtk_scrolled_window_new(NULL, NULL); + gtk_widget_show(scrolledwindow1); + gtk_box_pack_start(GTK_BOX(hbox3), scrolledwindow1, TRUE, TRUE, 0); + gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolledwindow1), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); + gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolledwindow1), GTK_SHADOW_IN); + + + + + + + store = gtk_tree_store_new(PTC_COUNT, G_TYPE_BOOLEAN, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING); + g_object_set_data(ref, "store", store); + + treeview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(store)); + gtk_widget_show(treeview); + gtk_container_add(GTK_CONTAINER(scrolledwindow1), treeview); + + g_object_unref(G_OBJECT(store)); + /* + column = gtk_tree_view_column_new(); + gtk_tree_view_column_set_visible(column, FALSE); + gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column); + gtk_tree_view_set_expander_column(GTK_TREE_VIEW(treeview), column); + */ + + renderer = gtk_cell_renderer_toggle_new(); + gtk_cell_renderer_toggle_set_activatable(GTK_CELL_RENDERER_TOGGLE(renderer), TRUE); + column = gtk_tree_view_column_new_with_attributes(_("Active"), renderer, "active", PTC_ACTIVE, NULL); + gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column); + g_signal_connect(G_OBJECT(renderer), "toggled", G_CALLBACK(on_part_selection_toggle), ref); + + renderer = gtk_cell_renderer_text_new(); + column = gtk_tree_view_column_new_with_attributes(_("Name"), renderer, "text", PTC_NAME, NULL); + gtk_tree_view_column_set_expand(column, TRUE); + gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column); + + renderer = gtk_cell_renderer_text_new(); + column = gtk_tree_view_column_new_with_attributes(_("Start"), renderer, "text", PTC_START, 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(_("End"), renderer, "text", PTC_END, NULL); + gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column); + + + + + + + + vbuttonbox1 = gtk_vbutton_box_new(); + gtk_widget_show(vbuttonbox1); + gtk_box_pack_start(GTK_BOX(hbox3), vbuttonbox1, FALSE, FALSE, 0); + gtk_button_box_set_layout(GTK_BUTTON_BOX(vbuttonbox1), GTK_BUTTONBOX_SPREAD); + + + + + button = qck_create_button_with_img(NULL, NULL, "gtk-add", G_CALLBACK(NULL), NULL); + gtk_container_add(GTK_CONTAINER(vbuttonbox1), button); + + button = qck_create_button_with_img(NULL, NULL, "gtk-remove", G_CALLBACK(NULL), NULL); + gtk_container_add(GTK_CONTAINER(vbuttonbox1), button); + + + + + + hbuttonbox1 = gtk_hbutton_box_new(); + gtk_widget_show(hbuttonbox1); + gtk_box_pack_start(GTK_BOX(vbox1), hbuttonbox1, FALSE, FALSE, 0); + gtk_button_box_set_layout(GTK_BUTTON_BOX(hbuttonbox1), GTK_BUTTONBOX_END); + + + button = qck_create_button_from_stock(NULL, NULL, "gtk-ok", G_CALLBACK(NULL), NULL); + gtk_container_add(GTK_CONTAINER(hbuttonbox1), button); + + button = qck_create_button_from_stock(NULL, NULL, "gtk-cancel", G_CALLBACK(NULL), NULL); + gtk_container_add(GTK_CONTAINER(hbuttonbox1), button); + + + + + + + g_signal_connect(G_OBJECT(comboboxentry), "changed", G_CALLBACK(on_model_change), ref); + + gtk_combo_box_append_text(GTK_COMBO_BOX(comboboxentry), _("Default")); + gtk_combo_box_append_text(GTK_COMBO_BOX(comboboxentry), _("Routines")); + gtk_combo_box_append_text(GTK_COMBO_BOX(comboboxentry), _("User")); + + return result; + +} + + + + + + + +/****************************************************************************** +* * +* Paramètres : project = informations sur le project actuellement ouvert. * +* ref = espace de référencement principal. * +* * +* Description : Charge les sections sélectionnées pour le projet courant. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +void load_project_sections(void *project, GObject *ref) +{ + + + + /* + gtk_combo_box_append_text(GTK_COMBO_BOX(combobox), _("Never automatically")); + gtk_combo_box_append_text(GTK_COMBO_BOX(combobox), _("When the window gets the focus")); + gtk_combo_box_append_text(GTK_COMBO_BOX(combobox), _("On click on the window")); + + */ + +} + + + + + + + + + + +/****************************************************************************** +* * +* Paramètres : ref = espace de référencement principal. * +* * +* Description : Affiche les parties désassemblées par défaut. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void load_default_parts(GObject *ref) +{ + GOpenidaBinary *binary; /* Binaire à traiter */ + GtkTreeStore *store; /* Modèle de gestion */ + GExeFormat *format; /* Format associé au binaire */ + GArchProcessor *proc; /* Architecture utilisée */ + GBinPart **parts; /* Parties d'élément binaire */ + size_t parts_count; /* Nombre de ces parties */ + parts_model *model; /* Mémoire du modèle */ + bool exist; /* Mémoire présente ? */ + size_t i; /* Boucle de parcours */ + off_t size; /* Taille de la partie */ + vmpa_t addr; /* Adresse de départ */ + char start[VMPA_MAX_SIZE]; /* Version humainement lisible */ + char end[VMPA_MAX_SIZE]; /* Version humainement lisible */ + GtkTreeIter iter; /* Point d'insertion */ + + binary = G_OPENIDA_BINARY(g_object_get_data(ref, "binary")); + store = GTK_TREE_STORE(g_object_get_data(ref, "store")); + + format = g_openida_binary_get_format(binary); + proc = get_arch_processor_from_format(format); + + parts = g_exe_format_get_parts(format, &parts_count); + qsort(parts, parts_count, sizeof(GBinPart *), g_binary_part_compare); + + model = (parts_model *)g_object_get_data(ref, "default_model"); + exist = (model != NULL); + + if (!exist) + { + model = (parts_model *)calloc(1, sizeof(parts_model)); + g_object_set_data(ref, "default_model", model); + + model->selected = (gboolean *)calloc(parts_count, sizeof(gboolean)); + model->count = parts_count; + + } + + for (i = 0; i < parts_count; i++) + { + g_binary_part_get_values(parts[i], NULL, &size, &addr); + + vmpa_to_string(addr, g_arch_processor_get_memory_size(proc), start); + vmpa_to_string(addr + size, g_arch_processor_get_memory_size(proc), end); + + if (!exist) + model->selected[i] = TRUE; + + gtk_tree_store_append(store, &iter, NULL); + gtk_tree_store_set(store, &iter, + PTC_ACTIVE, model->selected[i], + PTC_NAME, g_binary_part_get_name(parts[i]), + PTC_START, start, + PTC_END, end, + -1); + + } + +} + + +/****************************************************************************** +* * +* Paramètres : ref = espace de référencement principal. * +* * +* Description : Affiche les parties désassemblées selon les routines. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void load_routines_parts(GObject *ref) +{ + GOpenidaBinary *binary; /* Binaire à traiter */ + GtkTreeStore *store; /* Modèle de gestion */ + GExeFormat *format; /* Format associé au binaire */ + GArchProcessor *proc; /* Architecture utilisée */ + GBinRoutine **routines; /* Liste des routines trouvées */ + size_t routines_count; /* Nombre de ces routines */ + parts_model *model; /* Mémoire du modèle */ + bool exist; /* Mémoire présente ? */ + size_t i; /* Boucle de parcours */ + vmpa_t addr; /* Adresse à transcrire */ + char start[VMPA_MAX_SIZE]; /* Version humainement lisible */ + char end[VMPA_MAX_SIZE]; /* Version humainement lisible */ + GtkTreeIter iter; /* Point d'insertion */ + + binary = G_OPENIDA_BINARY(g_object_get_data(ref, "binary")); + store = GTK_TREE_STORE(g_object_get_data(ref, "store")); + + format = g_openida_binary_get_format(binary); + proc = get_arch_processor_from_format(format); + + routines = g_binary_format_get_routines(G_BIN_FORMAT(format), &routines_count); + qsort(routines, routines_count, sizeof(GBinRoutine *), g_binary_routine_compare); + + model = (parts_model *)g_object_get_data(ref, "routines_model"); + exist = (model != NULL); + + if (!exist) + { + model = (parts_model *)calloc(1, sizeof(parts_model)); + g_object_set_data(ref, "routines_model", model); + + model->selected = (gboolean *)calloc(routines_count, sizeof(gboolean)); + model->count = routines_count; + + } + + for (i = 0; i < routines_count; i++) + { + addr = g_binary_routine_get_address(routines[i]); + vmpa_to_string(addr, g_arch_processor_get_memory_size(proc), start); + + addr += g_binary_routine_get_size(routines[i]); + vmpa_to_string(addr, g_arch_processor_get_memory_size(proc), end); + + if (!exist) + model->selected[i] = TRUE; + + gtk_tree_store_append(store, &iter, NULL); + gtk_tree_store_set(store, &iter, + PTC_ACTIVE, model->selected[i], + PTC_NAME, g_binary_routine_get_name(routines[i]), + PTC_START, start, + PTC_END, end, + -1); + + } + +} + + + + + +static void user_function(GtkButton *button, GObject *ref) +{ + + + + + + + + +} + + + + + + + + +/****************************************************************************** +* * +* Paramètres : combo = liste des modèles proposés. * +* ref = espace de référencement principal. * +* * +* Description : Réagit à un changement de modèle. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void on_model_change(GtkComboBox *combo, GObject *ref) +{ + gint index; /* Indice du nouveau modèle */ + GtkTreeStore *store; /* Modèle de gestion */ + + store = GTK_TREE_STORE(g_object_get_data(ref, "store")); + gtk_tree_store_clear(store); + + index = gtk_combo_box_get_active(combo); + + switch (index) + { + case BPM_DEFAULT: + load_default_parts(ref); + break; + + case BPM_ROUTINES: + load_routines_parts(ref); + break; + + case BPM_USER: + break; + + } + +} + + +/****************************************************************************** +* * +* Paramètres : renderer = cellule de rendu à mettre à jour. * +* path = chemin menant à la ligne concernée. * +* ref = espace de référencement principal. * +* * +* Description : Réagit à un changement de sélection de partie. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void on_part_selection_toggle(GtkCellRendererToggle *renderer, gchar *path, GObject *ref) +{ + GtkTreeModel *model; /* Modèle de représentation */ + GtkTreeIter iter; /* Lieu de la mise à jour */ + gboolean state; /* Etat de la sélection */ + GtkComboBox *combo; /* Liste de tous les modèles */ + gint index; /* Indice du modèle courant */ + parts_model *list; /* Mémorisation des sélections */ + + + printf("path :: %s\n", path); + + model = GTK_TREE_MODEL(g_object_get_data(ref, "store")); + + if (gtk_tree_model_get_iter_from_string(model, &iter, path)) + { + gtk_tree_model_get(model, &iter, PTC_ACTIVE, &state, -1); + + combo = GTK_COMBO_BOX(g_object_get_data(ref, "models")); + index = gtk_combo_box_get_active(combo); + + switch (index) + { + case BPM_DEFAULT: + list = (parts_model *)g_object_get_data(ref, "default_model"); + break; + + case BPM_ROUTINES: + list = (parts_model *)g_object_get_data(ref, "routines_model"); + break; + + case BPM_USER: + break; + + } + + list->selected[atoi(path)] = !state; + + gtk_tree_store_set(GTK_TREE_STORE(model), &iter, + PTC_ACTIVE, !state, + -1); + + } + +} diff --git a/src/dialogs/binparts.h b/src/dialogs/binparts.h new file mode 100644 index 0000000..dd7fa5e --- /dev/null +++ b/src/dialogs/binparts.h @@ -0,0 +1,42 @@ + +/* OpenIDA - Outil d'analyse de fichiers binaires + * dlg_sections.h - prototypes pour la boîte de dialogue permettant une sélection des sections + * + * Copyright (C) 2008 Cyrille Bagard + * + * This file is part of OpenIDA. + * + * OpenIDA 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. + * + * OpenIDA 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 Foobar. If not, see . + */ + + +#ifndef _DLG_SECTIONS_H +#define _DLG_SECTIONS_H + + +#include + + +#include "../analysis/binary.h" + + + +/* Construit la fenêtre de sélection des sections. */ +GtkWidget *create_sections_dialog(GOpenidaBinary *); + + + + + +#endif /* _DLG_SECTIONS_H */ diff --git a/src/dlg_sections.c b/src/dlg_sections.c index 44a00a7..e69de29 100644 --- a/src/dlg_sections.c +++ b/src/dlg_sections.c @@ -1,218 +0,0 @@ - -/* OpenIDA - Outil d'analyse de fichiers binaires - * dlg_sections.h - boîte de dialogue permettant une sélection des sections - * - * Copyright (C) 2008 Cyrille Bagard - * - * This file is part of OpenIDA. - * - * OpenIDA 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. - * - * OpenIDA 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 Foobar. If not, see . - */ - - -#include "dlg_sections.h" - - -#include "gtkext/easygtk.h" - - - -#define _(str) str - - - - -/* Charge les sections sélectionnées pour le projet courant. */ -void load_project_sections(openida_project *, GObject *); - - - - -/****************************************************************************** -* * -* Paramètres : project = informations sur le project actuellement ouvert. * -* * -* Description : Construit la fenêtre de sélection des sections. * -* * -* Retour : Adresse de la fenêtre mise en place. * -* * -* Remarques : - * -* * -******************************************************************************/ - -GtkWidget *create_sections_dialog(openida_project *project) -{ - GtkWidget *result; /* Fenêtre à renvoyer */ - - - - - - GtkWidget *vbox1; - GtkWidget *hbox1; - - - GtkWidget *label; /* Etiquette à afficher */ - - - GtkWidget *combobox1; - - GtkWidget *alignment; /* Adaptation de disposition */ - GtkWidget *frame; /* Support avec encadrement */ - - GtkWidget *vbox2; - GtkWidget *hbox2; - GtkWidget *comboboxentry1; - GtkWidget *hbox3; - GtkWidget *scrolledwindow1; - GtkWidget *treeview1; - GtkWidget *vbuttonbox1; - GtkWidget *hbuttonbox1; - - GtkWidget *button; /* Bouton de commande */ - - result = gtk_window_new(GTK_WINDOW_TOPLEVEL); - gtk_widget_set_size_request(result, 400, 300); - gtk_container_set_border_width(GTK_CONTAINER(result), 8); - gtk_window_set_title(GTK_WINDOW(result), _("Sections selection")); - gtk_window_set_position(GTK_WINDOW(result), GTK_WIN_POS_CENTER); - gtk_window_set_default_size(GTK_WINDOW(result), 500, 400); - gtk_window_set_type_hint(GTK_WINDOW(result), GDK_WINDOW_TYPE_HINT_DIALOG); - - vbox1 = gtk_vbox_new(FALSE, 8); - gtk_widget_show(vbox1); - gtk_container_add(GTK_CONTAINER(result), vbox1); - - hbox1 = gtk_hbox_new(FALSE, 8); - gtk_widget_show(hbox1); - gtk_box_pack_start(GTK_BOX(vbox1), hbox1, FALSE, FALSE, 0); - - label = qck_create_label(NULL, NULL, _("Binary :")); - gtk_box_pack_start(GTK_BOX(hbox1), label, FALSE, FALSE, 0); - - combobox1 = gtk_combo_box_new_text(); - gtk_widget_show(combobox1); - gtk_box_pack_start(GTK_BOX(hbox1), combobox1, TRUE, TRUE, 0); - - - frame = qck_create_frame(_("Content to display"), &alignment, 4, 4, 12, 0); - gtk_box_pack_start(GTK_BOX(vbox1), frame, TRUE, TRUE, 0); - - - - - vbox2 = gtk_vbox_new(FALSE, 8); - gtk_widget_show(vbox2); - gtk_container_add(GTK_CONTAINER(alignment), vbox2); - - hbox2 = gtk_hbox_new(FALSE, 8); - gtk_widget_show(hbox2); - gtk_box_pack_start(GTK_BOX(vbox2), hbox2, FALSE, FALSE, 0); - - label = qck_create_label(NULL, NULL, _("Model :")); - gtk_box_pack_start(GTK_BOX(hbox2), label, FALSE, FALSE, 0); - - comboboxentry1 = qck_create_combobox(NULL, NULL, G_CALLBACK(NULL), NULL); - gtk_box_pack_start(GTK_BOX(hbox2), comboboxentry1, TRUE, TRUE, 0); - - - button = qck_create_button_with_img(NULL, NULL, "gtk-add", G_CALLBACK(NULL), NULL); - gtk_box_pack_start(GTK_BOX(hbox2), button, FALSE, FALSE, 0); - - button = qck_create_button_with_img(NULL, NULL, "gtk-remove", G_CALLBACK(NULL), NULL); - gtk_box_pack_start(GTK_BOX(hbox2), button, FALSE, FALSE, 0); - - - - - hbox3 = gtk_hbox_new(FALSE, 8); - gtk_widget_show(hbox3); - gtk_box_pack_start(GTK_BOX(vbox2), hbox3, TRUE, TRUE, 0); - - scrolledwindow1 = gtk_scrolled_window_new(NULL, NULL); - gtk_widget_show(scrolledwindow1); - gtk_box_pack_start(GTK_BOX(hbox3), scrolledwindow1, TRUE, TRUE, 0); - gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolledwindow1), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); - gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolledwindow1), GTK_SHADOW_IN); - - treeview1 = gtk_tree_view_new(); - gtk_widget_show(treeview1); - gtk_container_add(GTK_CONTAINER(scrolledwindow1), treeview1); - - vbuttonbox1 = gtk_vbutton_box_new(); - gtk_widget_show(vbuttonbox1); - gtk_box_pack_start(GTK_BOX(hbox3), vbuttonbox1, FALSE, FALSE, 0); - gtk_button_box_set_layout(GTK_BUTTON_BOX(vbuttonbox1), GTK_BUTTONBOX_SPREAD); - - - - - button = qck_create_button_with_img(NULL, NULL, "gtk-add", G_CALLBACK(NULL), NULL); - gtk_container_add(GTK_CONTAINER(vbuttonbox1), button); - - button = qck_create_button_with_img(NULL, NULL, "gtk-remove", G_CALLBACK(NULL), NULL); - gtk_container_add(GTK_CONTAINER(vbuttonbox1), button); - - - - - - hbuttonbox1 = gtk_hbutton_box_new(); - gtk_widget_show(hbuttonbox1); - gtk_box_pack_start(GTK_BOX(vbox1), hbuttonbox1, FALSE, FALSE, 0); - gtk_button_box_set_layout(GTK_BUTTON_BOX(hbuttonbox1), GTK_BUTTONBOX_END); - - - button = qck_create_button_from_stock(NULL, NULL, "gtk-ok", G_CALLBACK(NULL), NULL); - gtk_container_add(GTK_CONTAINER(hbuttonbox1), button); - - button = qck_create_button_from_stock(NULL, NULL, "gtk-cancel", G_CALLBACK(NULL), NULL); - gtk_container_add(GTK_CONTAINER(hbuttonbox1), button); - - return result; - -} - - - - - - - -/****************************************************************************** -* * -* Paramètres : project = informations sur le project actuellement ouvert. * -* ref = espace de référencement principal. * -* * -* Description : Charge les sections sélectionnées pour le projet courant. * -* * -* Retour : - * -* * -* Remarques : - * -* * -******************************************************************************/ - -void load_project_sections(openida_project *project, GObject *ref) -{ - - - - /* - gtk_combo_box_append_text(GTK_COMBO_BOX(combobox), _("Never automatically")); - gtk_combo_box_append_text(GTK_COMBO_BOX(combobox), _("When the window gets the focus")); - gtk_combo_box_append_text(GTK_COMBO_BOX(combobox), _("On click on the window")); - - */ - -} diff --git a/src/dlg_sections.h b/src/dlg_sections.h index 2541d2b..e69de29 100644 --- a/src/dlg_sections.h +++ b/src/dlg_sections.h @@ -1,42 +0,0 @@ - -/* OpenIDA - Outil d'analyse de fichiers binaires - * dlg_sections.h - prototypes pour la boîte de dialogue permettant une sélection des sections - * - * Copyright (C) 2008 Cyrille Bagard - * - * This file is part of OpenIDA. - * - * OpenIDA 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. - * - * OpenIDA 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 Foobar. If not, see . - */ - - -#ifndef _DLG_SECTIONS_H -#define _DLG_SECTIONS_H - - -#include - - -#include "project.h" - - - -/* Construit la fenêtre de sélection des sections. */ -GtkWidget *create_sections_dialog(openida_project *); - - - - - -#endif /* _DLG_SECTIONS_H */ diff --git a/src/editor.c b/src/editor.c index 5cdd4df..eed0868 100644 --- a/src/editor.c +++ b/src/editor.c @@ -39,8 +39,8 @@ /** exemple GTK **/ +#include "project.h" -#include "dlg_sections.h" #include "analysis/binary.h" #include "gtkext/easygtk.h" #include "gtkext/gtkextstatusbar.h" @@ -49,6 +49,7 @@ #include "gtkext/gtkdockpanel.h" #include "debug/debuggers.h" +#include "dialogs/binparts.h" #include "panels/panel.h" @@ -106,15 +107,16 @@ void mcb_project_remove_binary(GtkMenuItem *, gpointer); -/* Affiche la boîte de sélection des sections. */ -void mcb_select_sections(GtkMenuItem *, gpointer); - /* Met à jour le contenu du menu 'Projet'. */ void reload_menu_project(GObject *); +/* Réagit avec le menu "Binaire -> Sélectionner les parties...". */ +static void mcb_binary_select_parts(GtkMenuItem *, GObject *); + + /*Réagit avec le menu "Débogage -> Démarrer". */ void mcb_debug_start(GtkCheckMenuItem *, gpointer); @@ -309,7 +311,16 @@ GtkWidget *create_editor(void) submenuitem = qck_create_menu_separator(); gtk_container_add(GTK_CONTAINER(menubar), submenuitem); - submenuitem = qck_create_menu_item(NULL, NULL, _("Select sections..."), G_CALLBACK(mcb_select_sections), NULL); + + + menuitem = gtk_menu_item_new_with_mnemonic(_("_Binary")); + gtk_widget_show(menuitem); + gtk_container_add(GTK_CONTAINER(menuboard), menuitem); + + menubar = gtk_menu_new(); + gtk_menu_item_set_submenu(GTK_MENU_ITEM(menuitem), menubar); + + submenuitem = qck_create_menu_item(NULL, NULL, _("Select parts..."), G_CALLBACK(mcb_binary_select_parts), ref); gtk_container_add(GTK_CONTAINER(menubar), submenuitem); @@ -932,28 +943,6 @@ void mcb_project_remove_binary(GtkMenuItem *menuitem, gpointer data) -/****************************************************************************** -* * -* Paramètres : menuitem = élément de menu sélectionné. * -* data = adresse de l'espace de référencement global. * -* * -* Description : Affiche la boîte de sélection des sections. * -* * -* Retour : - * -* * -* Remarques : - * -* * -******************************************************************************/ - -void mcb_select_sections(GtkMenuItem *menuitem, gpointer data) -{ - GtkWidget *dialog; /* Boîte de dialogue à montrer */ - - dialog = create_sections_dialog(create_empty_openida_project(G_OBJECT(data))/* FIXME */); - gtk_widget_show(dialog); - -} - @@ -1016,6 +1005,32 @@ void reload_menu_project(GObject *ref) +/****************************************************************************** +* * +* Paramètres : menuitem = élément de menu sélectionné. * +* ref = adresse de l'espace de référencement global. * +* * +* Description : Réagit avec le menu "Binaire -> Sélectionner les parties...".* +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void mcb_binary_select_parts(GtkMenuItem *menuitem, GObject *ref) +{ + GOpenidaBinary *binary; /* Binaire courant à l'écran */ + GtkWidget *dialog; /* Boîte de dialogue à montrer */ + + binary = G_OPENIDA_BINARY(g_object_get_data(ref, "current_binary")); + + dialog = create_sections_dialog(binary); + gtk_widget_show(dialog); + +} + + void debugger_stopped_cb(GBinaryDebugger *debugger, uint64_t last, uint64_t cur, gpointer data) { diff --git a/src/format/elf/elf.c b/src/format/elf/elf.c index 34c7521..f32bce7 100644 --- a/src/format/elf/elf.c +++ b/src/format/elf/elf.c @@ -407,14 +407,19 @@ static GBinPart **g_elf_format_get_parts(const GElfFormat *format, size_t *count { GBinPart **result; /* Tableau à retourner */ uint16_t i; /* Boucle de parcours */ + elf_shdr strings; /* Section des descriptions */ + bool has_strings; /* Section trouvée ? */ elf_shdr section; /* En-tête de section ELF */ GBinPart *part; /* Partie à intégrer à la liste*/ + const char *name; /* Nom trouvé ou NULL */ off_t offset; /* Début de part de programme */ elf_phdr phdr; /* En-tête de programme ELF */ result = NULL; *count = 0; + has_strings = find_elf_section_by_index(format, format->header.e_shstrndx, &strings); + /* Première tentative : les sections */ for (i = 0; i < format->header.e_shnum; i++) @@ -427,7 +432,15 @@ static GBinPart **g_elf_format_get_parts(const GElfFormat *format, size_t *count { part = g_binary_part_new(); - /* TODO : nom, droits/type */ + if (has_strings) + { + name = extract_name_from_elf_string_section(format, &strings, + ELF_SHDR(format, section, sh_name)); + + if (name != NULL) + g_binary_part_set_name(part, name); + + } g_binary_part_set_values(part, ELF_SHDR(format, section, sh_offset), diff --git a/src/format/elf/section.c b/src/format/elf/section.c index e019648..f837dbf 100644 --- a/src/format/elf/section.c +++ b/src/format/elf/section.c @@ -92,6 +92,8 @@ bool find_elf_section_by_name(const GElfFormat *format, const char *name, elf_sh secname = extract_name_from_elf_string_section(format, &strings, ELF_SHDR(format, *section, sh_name)); + /* FIXME : if secname == NULL */ + result = (strcmp(name, secname) == 0); } diff --git a/src/format/part.c b/src/format/part.c index 2096c6d..d21cd87 100644 --- a/src/format/part.c +++ b/src/format/part.c @@ -145,6 +145,25 @@ void g_binary_part_set_name(GBinPart *part, const char *name) /****************************************************************************** * * +* Paramètres : part = description de partie à mettre à jour. * +* * +* Description : Fournit la description attribuée à une partie de code. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +const char *g_binary_part_get_name(const GBinPart *part) +{ + return part->name; + +} + + +/****************************************************************************** +* * * Paramètres : part = description de partie à mettre à jour. * * offset = position de la section à conserver. * * size = taille de la section à conserver. * diff --git a/src/format/part.h b/src/format/part.h index aec324d..ff18fc9 100644 --- a/src/format/part.h +++ b/src/format/part.h @@ -57,6 +57,9 @@ GBinPart *g_binary_part_new(void); /* Attribue une description humaine à une partie de code. */ void g_binary_part_set_name(GBinPart *, const char *); +/* Fournit la description attribuée à une partie de code. */ +const char *g_binary_part_get_name(const GBinPart *); + /* Définit les valeurs utiles d'une partie de code. */ void g_binary_part_set_values(GBinPart *, off_t, off_t, vmpa_t); diff --git a/src/project.c b/src/project.c index 2b70a47..1366abf 100644 --- a/src/project.c +++ b/src/project.c @@ -402,9 +402,9 @@ bool g_openida_project_save(openida_project *project, const char *filename) /* Sauvegarde finale */ - result &= save_xml_file(xdoc, filename); + result &= save_xml_file(xdoc, filename != NULL ? filename : project->filename); - if (result) + if (result && filename != NULL) { if (project->filename != NULL) free(project->filename); project->filename = strdup(filename); -- cgit v0.11.2-87-g4458