From c301f7e77eaca632a491b5b4417c8e4b9dce2570 Mon Sep 17 00:00:00 2001 From: Cyrille Bagard <nocbos@gmail.com> Date: Sat, 9 Nov 2019 14:18:35 +0100 Subject: Introduced the first features of a hexadecimal viewer. --- configure.ac | 3 +- src/analysis/binary.c | 24 +- src/analysis/binary.h | 1 + src/glibext/gbinarycursor.c | 57 +++- src/glibext/gbinarycursor.h | 3 + src/glibext/gbuffercache.c | 2 +- src/glibext/generators/Makefile.am | 1 + src/glibext/generators/hex.c | 617 ++++++++++++++++++++++++++++++++++++ src/glibext/generators/hex.h | 67 ++++ src/glibext/linesegment.c | 1 + src/glibext/linesegment.h | 1 + src/gtkext/Makefile.am | 1 + src/gtkext/gtkbufferdisplay.c | 2 + src/gtkext/gtkstatusstack.c | 40 +-- src/gtkext/gtkstatusstack.h | 4 +- src/gtkext/hexdisplay.c | 332 +++++++++++++++++++ src/gtkext/hexdisplay.h | 59 ++++ src/gui/menus/edition.c | 5 +- themes/Adwaita/display.css | 45 --- themes/Makefile.am | 4 +- themes/Solarized/Makefile.am | 18 ++ themes/Solarized/definition.xml | 22 ++ themes/Solarized/gresource.xml | 12 + themes/Solarized/palette.css | 17 + themes/Solarized/segments-dark.css | 61 ++++ themes/Solarized/segments-light.css | 61 ++++ themes/Solarized/segments.css | 48 +++ themes/Solarized/widgets-dark.css | 6 + themes/Solarized/widgets-light.css | 6 + 29 files changed, 1440 insertions(+), 80 deletions(-) create mode 100644 src/glibext/generators/hex.c create mode 100644 src/glibext/generators/hex.h create mode 100644 src/gtkext/hexdisplay.c create mode 100644 src/gtkext/hexdisplay.h create mode 100644 themes/Solarized/Makefile.am create mode 100644 themes/Solarized/definition.xml create mode 100644 themes/Solarized/gresource.xml create mode 100644 themes/Solarized/palette.css create mode 100644 themes/Solarized/segments-dark.css create mode 100644 themes/Solarized/segments-light.css create mode 100644 themes/Solarized/segments.css create mode 100644 themes/Solarized/widgets-dark.css create mode 100644 themes/Solarized/widgets-light.css diff --git a/configure.ac b/configure.ac index e58a0d6..e0e3042 100644 --- a/configure.ac +++ b/configure.ac @@ -508,7 +508,8 @@ AC_CONFIG_FILES([Makefile tools/d2c/pattern/Makefile tools/d2c/rules/Makefile themes/Makefile - themes/Adwaita/Makefile]) + themes/Adwaita/Makefile + themes/Solarized/Makefile]) AC_OUTPUT diff --git a/src/analysis/binary.c b/src/analysis/binary.c index 61dbbf1..1ba54e6 100644 --- a/src/analysis/binary.c +++ b/src/analysis/binary.c @@ -53,6 +53,7 @@ #include "../gtkext/gtkblockdisplay.h" #include "../gtkext/gtkdisplaypanel.h" #include "../gtkext/gtkgraphdisplay.h" +#include "../gtkext/hexdisplay.h" @@ -255,6 +256,10 @@ static void g_loaded_binary_init(GLoadedBinary *binary) binary->collections = create_collections_list(); attach_binary_to_collections(binary->collections, binary); + binary->options[BVW_HEX] = g_display_options_new(); + + g_display_options_add(binary->options[BVW_HEX], _("Physical offset"), true); + binary->options[BVW_BLOCK] = g_display_options_new(); g_display_options_add(binary->options[BVW_BLOCK], _("Physical offset"), true); @@ -1685,6 +1690,7 @@ static bool g_loaded_binary_analyze(GLoadedBinary *binary, bool connect, bool ca has_virt = g_arch_processor_has_virtual_space(binary->proc); + g_display_options_set(binary->options[BVW_HEX], 0, false); g_display_options_set(binary->options[BVW_BLOCK], BLC_VIRTUAL, has_virt); g_display_options_set(binary->options[BVW_GRAPH], BLC_VIRTUAL, has_virt); @@ -1884,6 +1890,10 @@ static const char *g_loaded_binary_get_view_name(const GLoadedBinary *binary, un switch (index) { + case BVW_HEX: + result = _("Hex view"); + break; + case BVW_BLOCK: result = _("Text view"); break; @@ -1919,12 +1929,19 @@ static const char *g_loaded_binary_get_view_name(const GLoadedBinary *binary, un static GtkWidget *g_loaded_binary_build_view(GLoadedBinary *binary, unsigned int index) { GtkWidget *result; /* Support à retourner */ + GBinContent *content; /* Contenu à représenter */ + GtkWidget *display; /* Composant d'affichage */ GBufferCache *cache; /* Tampon par défaut */ GBufferView *view; /* Vue sur ce même tampon */ - GtkWidget *display; /* Composant d'affichage */ switch (index) { + case BVW_HEX: + content = g_binary_format_get_content(G_BIN_FORMAT(binary->format)); + display = gtk_hex_display_new(content); + g_object_unref(G_OBJECT(content)); + break; + case BVW_BLOCK: cache = g_loaded_binary_get_disassembled_cache(binary); view = g_buffer_view_new(cache, NULL); @@ -1970,7 +1987,10 @@ static unsigned int g_loaded_binary_get_view_index(GLoadedBinary *binary, GtkWid { unsigned int result; /* Indice à retourner */ - if (GTK_IS_BLOCK_DISPLAY(view)) + if (GTK_IS_HEX_DISPLAY(view)) + result = BVW_HEX; + + else if (GTK_IS_BLOCK_DISPLAY(view)) result = BVW_BLOCK; else if (GTK_IS_GRAPH_DISPLAY(view)) diff --git a/src/analysis/binary.h b/src/analysis/binary.h index 1f94e34..65291f1 100644 --- a/src/analysis/binary.h +++ b/src/analysis/binary.h @@ -163,6 +163,7 @@ vmpa2t *g_loaded_binary_get_old_gotos(GLoadedBinary *, size_t *); /* Type de représentations */ typedef enum _BinaryView { + BVW_HEX, /* Contenu en hexadécimal */ BVW_BLOCK, /* Version basique */ BVW_GRAPH, /* Affichage en graphique */ diff --git a/src/glibext/gbinarycursor.c b/src/glibext/gbinarycursor.c index 9c4e82b..86f931f 100644 --- a/src/glibext/gbinarycursor.c +++ b/src/glibext/gbinarycursor.c @@ -39,6 +39,7 @@ struct _GBinaryCursor { GLineCursor parent; /* A laisser en premier */ + bool raw; /* Position brute ? */ vmpa2t addr; /* Position mémoire du curseur */ }; @@ -162,6 +163,8 @@ static void g_binary_cursor_class_init(GBinaryCursorClass *class) static void g_binary_cursor_init(GBinaryCursor *cursor) { + cursor->raw = false; + init_vmpa(&cursor->addr, VMPA_NO_PHYSICAL, VMPA_NO_VIRTUAL); } @@ -246,6 +249,8 @@ static GLineCursor *g_binary_cursor_duplicate(const GBinaryCursor *cursor) result = g_binary_cursor_new(); + G_BINARY_CURSOR(result)->raw = cursor->raw; + g_binary_cursor_update(G_BINARY_CURSOR(cursor), &cursor->addr); return result; @@ -344,6 +349,9 @@ static void g_binary_cursor_show_status(const GBinaryCursor *cursor, GtkStatusSt { GLoadedBinary *binary; /* Binaire chargé et analysé */ GArchProcessor *proc; /* Architecture du binaire */ + mrange_t tmp; /* Emplacement réduit */ + const mrange_t *range; /* Emplacement d'instruction */ + const char *encoding; /* Encodage à présenter */ GArchInstruction *instr; /* Instruction présente */ if (g_binary_cursor_is_valid(cursor)) @@ -352,21 +360,56 @@ static void g_binary_cursor_show_status(const GBinaryCursor *cursor, GtkStatusSt binary = G_LOADED_BINARY(content); - proc = g_loaded_binary_get_processor(binary); + if (cursor->raw) + { + init_mrange(&tmp, &cursor->addr, VMPA_NO_PHYSICAL); + + gtk_status_stack_update_current_location(stack, binary, &tmp, NULL); + + } + + else + { + proc = g_loaded_binary_get_processor(binary); + + instr = _g_arch_processor_find_instr_by_address(proc, &cursor->addr, true); + assert(instr != NULL); - instr = _g_arch_processor_find_instr_by_address(proc, &cursor->addr, true); - assert(instr != NULL); + range = g_arch_instruction_get_range(instr); + encoding = g_arch_instruction_get_encoding(instr); - gtk_status_stack_update_current_instruction(stack, binary, instr); + gtk_status_stack_update_current_location(stack, binary, range, encoding); - g_object_unref(G_OBJECT(instr)); + g_object_unref(G_OBJECT(instr)); - g_object_unref(G_OBJECT(proc)); + g_object_unref(G_OBJECT(proc)); + + } } else - gtk_status_stack_reset_current_instruction(stack); + gtk_status_stack_reset_current_location(stack); + +} + + +/****************************************************************************** +* * +* Paramètres : cursor = suivi de positions à mettre à jour. * +* raw = nature de la représentation de l'emplacement visé. * +* * +* Description : Précise la représentation de l'emplacement. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +void g_binary_cursor_set_raw(GBinaryCursor *cursor, bool raw) +{ + cursor->raw = raw; } diff --git a/src/glibext/gbinarycursor.h b/src/glibext/gbinarycursor.h index ffac19a..defcb56 100644 --- a/src/glibext/gbinarycursor.h +++ b/src/glibext/gbinarycursor.h @@ -57,6 +57,9 @@ GType g_binary_cursor_get_type(void); /* Crée un nouveau suivi de positions dans un panneau. */ GLineCursor *g_binary_cursor_new(void); +/* Précise la représentation de l'emplacement. */ +void g_binary_cursor_set_raw(GBinaryCursor *, bool); + /* Met à jour la position suivi dans un panneau de chargement. */ void g_binary_cursor_update(GBinaryCursor *, const vmpa2t *); diff --git a/src/glibext/gbuffercache.c b/src/glibext/gbuffercache.c index 82d77c0..322631f 100644 --- a/src/glibext/gbuffercache.c +++ b/src/glibext/gbuffercache.c @@ -1321,7 +1321,7 @@ void g_buffer_cache_truncate(GBufferCache *cache, size_t max) removed = cache->used - max; - cache->used = max - 1; + cache->used = max; if (removed > 0) { diff --git a/src/glibext/generators/Makefile.am b/src/glibext/generators/Makefile.am index 2c3675f..ed1149e 100644 --- a/src/glibext/generators/Makefile.am +++ b/src/glibext/generators/Makefile.am @@ -2,6 +2,7 @@ noinst_LTLIBRARIES = libglibextgenerators.la libglibextgenerators_la_SOURCES = \ + hex.h hex.c \ prologue.h prologue.c \ rborder.h rborder.c diff --git a/src/glibext/generators/hex.c b/src/glibext/generators/hex.c new file mode 100644 index 0000000..48fc5b3 --- /dev/null +++ b/src/glibext/generators/hex.c @@ -0,0 +1,617 @@ + +/* Chrysalide - Outil d'analyse de fichiers binaires + * hex.c - génération à la volée de lignes hexadécimales + * + * Copyright (C) 2016 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 Foobar. If not, see <http://www.gnu.org/licenses/>. + */ + + +#include "hex.h" + + +#include <assert.h> +#include <ctype.h> + + +#include "../gbinarycursor.h" +#include "../gbufferline.h" +#include "../linegen-int.h" +#include "../linesegment.h" + + + +/* Tampon pour générateur de lignes hexadécimales (instance) */ +struct _GHexGenerator +{ + GObject parent; /* A laisser en premier */ + + GBinContent *content; /* Contenu à représenter */ + + gint left_start; /* Abscisse des impressions */ + + phys_t bytes_per_line; /* Nombre d'octets par ligne */ + +}; + +/* Tampon pour générateur de lignes hexadécimales (classe) */ +struct _GHexGeneratorClass +{ + GObjectClass parent; /* A laisser en premier */ + + gint addr_width; /* Largeur des positions */ + gint byte_width; /* Largeur d'un octet brut */ + gint sep_width; /* Largeur de séparation */ + gint char_width; /* Largeur d'un caractère */ + +}; + + +/* Procède à l'initialisation d'une classe de générateur. */ +static void g_hex_generator_class_init(GHexGeneratorClass *); + +/* Procède à l'initialisation d'un générateur de lignes hexa. */ +static void g_hex_generator_init(GHexGenerator *); + +/* Procède à l'initialisation de l'interface de génération. */ +static void g_hex_generator_interface_init(GLineGeneratorInterface *); + +/* Supprime toutes les références externes. */ +static void g_hex_generator_dispose(GHexGenerator *); + +/* Procède à la libération totale de la mémoire. */ +static void g_hex_generator_finalize(GHexGenerator *); + +/* Indique le nombre de ligne prêtes à être générées. */ +static size_t g_hex_generator_count_lines(const GHexGenerator *); + +/* Retrouve l'emplacement correspondant à une position donnée. */ +static void g_hex_generator_compute_cursor(const GHexGenerator *, gint, size_t, size_t, GLineCursor **); + +/* Détermine si le conteneur s'inscrit dans une plage donnée. */ +static int g_hex_generator_contains_cursor(const GHexGenerator *, size_t, size_t, const GLineCursor *); + +/* Renseigne sur les propriétés liées à un générateur. */ +static BufferLineFlags g_hex_generator_get_flags(const GHexGenerator *, size_t, size_t); + +/* Imprime dans une ligne de rendu le contenu représenté. */ +static void g_hex_generator_print(GHexGenerator *, GBufferLine *, size_t, size_t); + + + +/* Détermine le type du générateur de lignes hexadécimales à la volée. */ +G_DEFINE_TYPE_WITH_CODE(GHexGenerator, g_hex_generator, G_TYPE_OBJECT, + G_IMPLEMENT_INTERFACE(G_TYPE_LINE_GENERATOR, g_hex_generator_interface_init)); + + +/****************************************************************************** +* * +* Paramètres : class = classe de composant GLib à initialiser. * +* * +* Description : Procède à l'initialisation d'une classe de générateur. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void g_hex_generator_class_init(GHexGeneratorClass *class) +{ + GObjectClass *object; /* Autre version de la classe */ + line_segment *segment; /* Segment de test pour mesure */ + + object = G_OBJECT_CLASS(class); + + object->dispose = (GObjectFinalizeFunc/* ! */)g_hex_generator_dispose; + object->finalize = (GObjectFinalizeFunc)g_hex_generator_finalize; + + /* Mesure de quelques dimensions */ + + segment = get_new_line_segment(RTT_PHYS_ADDR, "0x00000000", 10); + + class->addr_width = get_line_segment_width(segment); + + release_line_segment(segment); + + segment = get_new_line_segment(RTT_RAW_CODE, "00", 2); + + class->byte_width = get_line_segment_width(segment); + + release_line_segment(segment); + + segment = get_new_line_segment(RTT_RAW_CODE, "\t", 1); + + class->sep_width = get_line_segment_width(segment); + + release_line_segment(segment); + + segment = get_new_line_segment(RTT_RAW_CODE, "0", 1); + + class->char_width = get_line_segment_width(segment); + + release_line_segment(segment); + +} + + +/****************************************************************************** +* * +* Paramètres : generator = composant GLib à initialiser. * +* * +* Description : Procède à l'initialisation d'un générateur de lignes hexa. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void g_hex_generator_init(GHexGenerator *generator) +{ + generator->bytes_per_line = 4; + +} + + +/****************************************************************************** +* * +* Paramètres : iface = interface GLib à initialiser. * +* * +* Description : Procède à l'initialisation de l'interface de génération. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void g_hex_generator_interface_init(GLineGeneratorInterface *iface) +{ + iface->count = (linegen_count_lines_fc)g_hex_generator_count_lines; + iface->compute = (linegen_compute_fc)g_hex_generator_compute_cursor; + iface->contains = (linegen_contains_fc)g_hex_generator_contains_cursor; + iface->get_flags = (linegen_get_flags_fc)g_hex_generator_get_flags; + iface->print = (linegen_print_fc)g_hex_generator_print; + +} + + +/****************************************************************************** +* * +* Paramètres : generator = instance d'objet GLib à traiter. * +* * +* Description : Supprime toutes les références externes. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void g_hex_generator_dispose(GHexGenerator *generator) +{ + if (generator->content != NULL) + g_object_unref(G_OBJECT(generator->content)); + + G_OBJECT_CLASS(g_hex_generator_parent_class)->dispose(G_OBJECT(generator)); + +} + + +/****************************************************************************** +* * +* Paramètres : generator = instance d'objet GLib à traiter. * +* * +* Description : Procède à la libération totale de la mémoire. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void g_hex_generator_finalize(GHexGenerator *generator) +{ + G_OBJECT_CLASS(g_hex_generator_parent_class)->finalize(G_OBJECT(generator)); + +} + + +/****************************************************************************** +* * +* Paramètres : content = contenu chargé à représenter partiellement. * +* * +* Description : Crée un nouveau générateur de lignes hexadécimales. * +* * +* Retour : Composant GLib créé. * +* * +* Remarques : - * +* * +******************************************************************************/ + +GHexGenerator *g_hex_generator_new(GBinContent *content) +{ + GHexGenerator *result; /* Composant à retourner */ + + result = g_object_new(G_TYPE_HEX_GENERATOR, NULL); + + result->content = content; + + g_object_ref(G_OBJECT(result->content)); + + return result; + +} + + +/****************************************************************************** +* * +* Paramètres : generator = générateur à consulter. * +* * +* Description : Indique le nombre de ligne prêtes à être générées. * +* * +* Retour : Nombre de lignes devant apparaître au final. * +* * +* Remarques : - * +* * +******************************************************************************/ + +static size_t g_hex_generator_count_lines(const GHexGenerator *generator) +{ + size_t result; /* Quantité à retourner */ + phys_t size; /* Taille du contenu binaire */ + + size = g_binary_content_compute_size(generator->content); + + result = size / generator->bytes_per_line; + + if (size % generator->bytes_per_line > 0) + result++; + + return result; + +} + + +/****************************************************************************** +* * +* Paramètres : generator = générateur à consulter. * +* x = position géographique sur la ligne concernée. * +* index = indice de cette même ligne dans le tampon global.* +* repeat = indice d'utilisations successives du générateur. * +* cursor = emplacement à constituer. [OUT] * +* * +* Description : Retrouve l'emplacement correspondant à une position donnée. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void g_hex_generator_compute_cursor(const GHexGenerator *generator, gint x, size_t index, size_t repeat, GLineCursor **cursor) +{ + phys_t phys; /* Position physique */ + GHexGeneratorClass *class; /* Accès aux informations */ + phys_t block_count; /* Nombre de blocs d'octets */ + gint bytes_width; /* Largeur des octets occupée */ + phys_t i; /* Boucle de parcours */ + gint requested; /* Espace requis pour X octets */ + vmpa2t pos; /* Position de la portion */ + + x -= generator->left_start; + + phys = generator->bytes_per_line * index; + + class = G_HEX_GENERATOR_GET_CLASS(generator); + + block_count = (generator->bytes_per_line / 4); + + bytes_width = generator->bytes_per_line * class->byte_width + + 3 * block_count * class->char_width + + (block_count > 1 ? block_count - 1 : 0) * class->sep_width; + + if (x <= bytes_width) + { + for (i = 1; i <= generator->bytes_per_line; i++) + { + block_count = (i / 4); + + requested = i * class->byte_width + (3 * block_count + ((i - 1) % 3)) * class->char_width; + requested += (block_count > 1 ? block_count - 1 : 0) * class->sep_width; + + if (requested > x) + break; + + phys++; + + } + + } + else + { + x -= (bytes_width + COL_MARGIN); + + phys += (x / class->char_width); + + } + + init_vmpa(&pos, phys, VMPA_NO_VIRTUAL); + + *cursor = g_binary_cursor_new(); + g_binary_cursor_set_raw(G_BINARY_CURSOR(*cursor), true); + + g_binary_cursor_update(G_BINARY_CURSOR(*cursor), &pos); + +} + + +/****************************************************************************** +* * +* Paramètres : generator = générateur à consulter. * +* index = indice de cette même ligne dans le tampon global.* +* repeat = indice d'utilisations successives du générateur. * +* cursor = emplacement à analyser. * +* * +* Description : Détermine si le conteneur s'inscrit dans une plage donnée. * +* * +* Retour : Bilan de la détermination, utilisable en comparaisons. * +* * +* Remarques : - * +* * +******************************************************************************/ + +static int g_hex_generator_contains_cursor(const GHexGenerator *generator, size_t index, size_t repeat, const GLineCursor *cursor) +{ + int result; /* Bilan d'étude à renvoyer */ + vmpa2t addr; /* Autre emplacement à comparer*/ + vmpa2t pos; /* Position de la portion */ + phys_t length; /* Quantité affichable */ + mrange_t range; /* Emplacement associé */ + + assert(G_IS_BINARY_CURSOR(cursor)); + + g_binary_cursor_get_info(G_BINARY_CURSOR(cursor), &addr); + + init_vmpa(&pos, generator->bytes_per_line * index, VMPA_NO_VIRTUAL); + + length = g_binary_content_compute_size(generator->content) - get_phy_addr(&pos); + + if (length > generator->bytes_per_line) + length = generator->bytes_per_line; + + init_mrange(&range, &pos, length); + + result = cmp_mrange_with_vmpa(&range, &addr); + + return result; + +} + + +/****************************************************************************** +* * +* Paramètres : generator = générateur à consulter. * +* index = indice de cette même ligne dans le tampon global.* +* repeat = indice d'utilisations successives du générateur. * +* * +* Description : Renseigne sur les propriétés liées à un générateur. * +* * +* Retour : Propriétés particulières associées. * +* * +* Remarques : - * +* * +******************************************************************************/ + +static BufferLineFlags g_hex_generator_get_flags(const GHexGenerator *generator, size_t index, size_t repeat) +{ + return BLF_HAS_CODE; + +} + + +/****************************************************************************** +* * +* Paramètres : generator = générateur à utiliser pour l'impression. * +* line = ligne de rendu à compléter. * +* index = indice de cette même ligne dans le tampon global.* +* repeat = indice d'utilisations successives du générateur. * +* * +* Description : Imprime dans une ligne de rendu le contenu représenté. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void g_hex_generator_print(GHexGenerator *generator, GBufferLine *line, size_t index, size_t repeat) +{ + vmpa2t pos; /* Position définie à la volée */ + phys_t got; /* Quantité affichable */ + const bin_t *raw; /* Accès direct et brut */ + phys_t i; /* Boucle de parcours */ + bin_t byte; /* Copie pour confort */ + char tmp[2]; /* Représentation d'un octet */ + + //static const char hexa[] = "0123456789abcdef"; + static const char hexa[] = "0123456789ABCDEF"; + + /* Position physique */ + + init_vmpa(&pos, generator->bytes_per_line * index, VMPA_NO_VIRTUAL); + + g_buffer_line_fill_phys(line, MDS_32_BITS_UNSIGNED, &pos); + + /* Contenu brut */ + + got = g_binary_content_compute_size(generator->content) - get_phy_addr(&pos); + + if (got > generator->bytes_per_line) + got = generator->bytes_per_line; + + raw = g_binary_content_get_raw_access(generator->content, &pos, got); + + for (i = 0; i < got; i++) + { + /* Séparation ? */ + + if (i > 0) + { + if (i % 4 == 0) + g_buffer_line_append_text(line, BLC_BINARY, "\t", 1, RTT_RAW, NULL); + else + g_buffer_line_append_text(line, BLC_BINARY, " ", 1, RTT_RAW, NULL); + } + + /* Binaire brut */ + + byte = raw[i]; + + if (byte == 0x00) + g_buffer_line_append_text(line, BLC_BINARY, "00", 2, RTT_RAW_NULL, NULL); + + else if (byte == 0xff) + g_buffer_line_append_text(line, BLC_BINARY, "ff", 2, RTT_RAW_FULL, NULL); + + else + { + tmp[1] = hexa[byte & 0xf]; + tmp[0] = hexa[(byte >> 4) & 0xf]; + + g_buffer_line_append_text(line, BLC_BINARY, tmp, 2, RTT_RAW, NULL); + + } + + /* Représentation humaine ? */ + + if (isgraph(byte) || byte == ' ') + g_buffer_line_append_text(line, BLC_ASSEMBLY, (char *)raw + i, 1, RTT_PRINTABLE, NULL); + else + g_buffer_line_append_text(line, BLC_ASSEMBLY, ".", 1, RTT_NOT_PRINTABLE, NULL); + + } + +} + + +/****************************************************************************** +* * +* Paramètres : generator = générateur à consulter. * +* * +* Description : Fournit le contenu associé au générateur de lignes hexa. * +* * +* Retour : Contenu dans lequel puise le générateur pour les lignes. * +* * +* Remarques : - * +* * +******************************************************************************/ + +GBinContent *g_hex_generator_get_content(const GHexGenerator *generator) +{ + GBinContent *result; /* Référence à retourner */ + + result = generator->content; + + g_object_ref(G_OBJECT(result)); + + return result; + +} + + +/****************************************************************************** +* * +* Paramètres : generator = générateur à ajuster. * +* left = position de départ à gauche pour le texte. * +* show_pos = affichage des positions sur la gauche ? * +* width = largeur maximale disponible. * +* * +* Description : Ajuste la génération à une nouvelle largeur de rendu. * +* * +* Retour : true si le nombre d'octets par ligne a changé, false sinon. * +* * +* Remarques : - * +* * +******************************************************************************/ + +bool g_hex_generator_auto_fit(GHexGenerator *generator, gint left, bool show_pos, gint width) +{ + bool result; /* Bilan à faire remonter */ + GHexGeneratorClass *class; /* Accès aux informations */ + phys_t i; /* Boucle de parcours */ + gint requested; /* Espace requis pour X octets */ + phys_t block_count; /* Nombre de blocs d'octets */ + + class = G_HEX_GENERATOR_GET_CLASS(generator); + + generator->left_start = left; + + if (show_pos) + generator->left_start += class->addr_width + COL_MARGIN; + + for (i = 4; ; i += 4) + { + requested = left; + + if (show_pos) + requested += class->addr_width + COL_MARGIN; + + block_count = (i / 4); + + requested += i * class->byte_width + 3 * block_count * class->char_width; + requested += (block_count > 1 ? block_count - 1 : 0) * class->sep_width; + requested += COL_MARGIN; + + requested += i * class->char_width; + + /* Limite atteinte ? */ + if (requested > width) + { + i -= 4; + break; + } + + } + + result = (generator->bytes_per_line != i); + + generator->bytes_per_line = i; + + return result; + +} + + +/****************************************************************************** +* * +* Paramètres : generator = générateur à consulter. * +* * +* Description : Indique le nombre d'octets programmés constituer une ligne. * +* * +* Retour : Nombre d'octets représentés sur chaque ligne. * +* * +* Remarques : - * +* * +******************************************************************************/ + +phys_t g_hex_generator_get_bytes_per_line(const GHexGenerator *generator) +{ + return generator->bytes_per_line; + +} diff --git a/src/glibext/generators/hex.h b/src/glibext/generators/hex.h new file mode 100644 index 0000000..4eb7b19 --- /dev/null +++ b/src/glibext/generators/hex.h @@ -0,0 +1,67 @@ + +/* Chrysalide - Outil d'analyse de fichiers binaires + * hex.h - prototypes pour la génération à la volée de lignes hexadécimales + * + * Copyright (C) 2016 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 Foobar. If not, see <http://www.gnu.org/licenses/>. + */ + + +#ifndef _GLIBEXT_GENERATORS_HEX_H +#define _GLIBEXT_GENERATORS_HEX_H + + +#include <glib-object.h> + + +#include "../../analysis/content.h" + + + +#define G_TYPE_HEX_GENERATOR (g_hex_generator_get_type()) +#define G_HEX_GENERATOR(obj) (G_TYPE_CHECK_INSTANCE_CAST((obj), G_TYPE_HEX_GENERATOR, GHexGenerator)) +#define G_IS_HEX_GENERATOR(obj) (G_TYPE_CHECK_INSTANCE_TYPE((obj), G_TYPE_HEX_GENERATOR)) +#define G_HEX_GENERATOR_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST((klass), G_TYPE_HEX_GENERATOR, GHexGeneratorClass)) +#define G_IS_HEX_GENERATOR_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE((klass), G_TYPE_HEX_GENERATOR)) +#define G_HEX_GENERATOR_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS((obj), G_TYPE_HEX_GENERATOR, GHexGeneratorClass)) + + +/* Tampon pour générateur de lignes hexadécimales (instance) */ +typedef struct _GHexGenerator GHexGenerator; + +/* Tampon pour générateur de lignes hexadécimales (classe) */ +typedef struct _GHexGeneratorClass GHexGeneratorClass; + + +/* Détermine le type du générateur de lignes hexadécimales à la volée. */ +GType g_hex_generator_get_type(void); + +/* Crée un nouveau générateur de lignes hexadécimales. */ +GHexGenerator *g_hex_generator_new(GBinContent *); + +/* Fournit le contenu associé au générateur de lignes hexa. */ +GBinContent *g_hex_generator_get_content(const GHexGenerator *); + +/* Ajuste la génération à une nouvelle largeur de rendu. */ +bool g_hex_generator_auto_fit(GHexGenerator *, gint, bool, gint); + +/* Indique le nombre d'octets programmés constituer une ligne. */ +phys_t g_hex_generator_get_bytes_per_line(const GHexGenerator *); + + + +#endif /* _GLIBEXT_GENERATORS_HEX_H */ diff --git a/src/glibext/linesegment.c b/src/glibext/linesegment.c index bd11afc..b769c01 100644 --- a/src/glibext/linesegment.c +++ b/src/glibext/linesegment.c @@ -51,6 +51,7 @@ static const char *_segment_names[RTT_COUNT] = { [RTT_NONE] = SEGMENT_NAME("none"), [RTT_RAW] = SEGMENT_NAME("raw"), + [RTT_RAW_FULL] = SEGMENT_NAME("raw-full"), [RTT_RAW_NULL] = SEGMENT_NAME("raw-null"), [RTT_PRINTABLE] = SEGMENT_NAME("printable"), [RTT_NOT_PRINTABLE] = SEGMENT_NAME("not-printable"), diff --git a/src/glibext/linesegment.h b/src/glibext/linesegment.h index 9723a80..e6a8fff 100644 --- a/src/glibext/linesegment.h +++ b/src/glibext/linesegment.h @@ -69,6 +69,7 @@ typedef enum _RenderingTagType RTT_NONE, /* Espace ou tabulation */ RTT_RAW, /* Contenu brut */ + RTT_RAW_FULL, /* Contenu brut et complet */ RTT_RAW_NULL, /* Contenu brut et nul */ RTT_PRINTABLE, /* Caractère imprimable */ RTT_NOT_PRINTABLE, /* Caractère non imprimable */ diff --git a/src/gtkext/Makefile.am b/src/gtkext/Makefile.am index 24df0c3..a56cf33 100644 --- a/src/gtkext/Makefile.am +++ b/src/gtkext/Makefile.am @@ -20,6 +20,7 @@ libgtkext_la_SOURCES = \ gtkdockstation.h gtkdockstation.c \ gtkgraphdisplay.h gtkgraphdisplay.c \ gtkstatusstack.h gtkstatusstack.c \ + hexdisplay.h hexdisplay.c \ rendering.h rendering.c \ resources.h resources.c \ support.h support.c \ diff --git a/src/gtkext/gtkbufferdisplay.c b/src/gtkext/gtkbufferdisplay.c index a1833a6..32a6a98 100644 --- a/src/gtkext/gtkbufferdisplay.c +++ b/src/gtkext/gtkbufferdisplay.c @@ -467,6 +467,8 @@ static gboolean gtk_buffer_display_draw(GtkWidget *widget, cairo_t *cr) gtk_style_context_add_class(context, GTK_STYLE_CLASS_VIEW); + gtk_style_context_add_class(context, "graph-block-background"); + gtk_render_background(context, cr, left_margin, area.y, area.width, area.height); gtk_style_context_restore(context); diff --git a/src/gtkext/gtkstatusstack.c b/src/gtkext/gtkstatusstack.c index 45bd667..843d444 100644 --- a/src/gtkext/gtkstatusstack.c +++ b/src/gtkext/gtkstatusstack.c @@ -125,7 +125,7 @@ static void on_size_allocate_for_asm_status(GtkWidget *, GdkRectangle *, GObject static void on_zoom_icon_press(GtkEntry *, GtkEntryIconPosition, GdkEventButton *, GtkStatusStack *); /* S'assure de l'affichage à jour de la partie "assemblage". */ -static gboolean gtk_status_stack_show_current_instruction(GtkStatusStack *); +static gboolean gtk_status_stack_show_current_location(GtkStatusStack *); @@ -229,7 +229,7 @@ static void gtk_status_stack_init(GtkStatusStack *stack) gtk_widget_show(GTK_WIDGET(stack->main)); gtk_box_pack_start(GTK_BOX(stack), GTK_WIDGET(stack->main), TRUE, TRUE, 8); - stack->def_source = (GSourceFunc)gtk_status_stack_show_current_instruction; + stack->def_source = (GSourceFunc)gtk_status_stack_show_current_location; layer = build_assembly_status_stack(stack); gtk_stack_add_named(stack->main, layer, "asm_info"); @@ -247,7 +247,7 @@ static void gtk_status_stack_init(GtkStatusStack *stack) reset_progress_info(stack->prog_info); - gtk_status_stack_reset_current_instruction(stack); + gtk_status_stack_reset_current_location(stack); } @@ -498,9 +498,10 @@ static void on_zoom_icon_press(GtkEntry *entry, GtkEntryIconPosition icon_pos, G /****************************************************************************** * * -* Paramètres : stack = barre de statut à actualiser. * -* binary = binaire chargé rassemblant l'ensemble des infos. * -* instr = instruction désassemblée ciblée graphiquement. * +* Paramètres : stack = barre de statut à actualiser. * +* binary = binaire chargé rassemblant l'ensemble des infos. * +* range = emplacement à mettre en valeur. * +* encoding = encodage d'une éventuelle instruction ou NULL. * * * * Description : Actualise les informations liées une position d'assemblage. * * * @@ -510,12 +511,12 @@ static void on_zoom_icon_press(GtkEntry *entry, GtkEntryIconPosition icon_pos, G * * ******************************************************************************/ -void gtk_status_stack_update_current_instruction(GtkStatusStack *stack, const GLoadedBinary *binary, const GArchInstruction *instr) +void gtk_status_stack_update_current_location(GtkStatusStack *stack, const GLoadedBinary *binary, const mrange_t *range, const char *encoding) { assembly_info *info; /* Informations à constituer */ GExeFormat *format; /* Format de binaire à traiter */ - const mrange_t *range; /* Emplacement d'instruction */ const vmpa2t *addr; /* Localisation de départ */ + phys_t size; /* Taille de l'emplacement */ GBinPortion *portions; /* Couche première de portions */ GBinPortion *portion; /* Zone mémoire d'appartenance */ const char *text; /* Texte au contenu à copier */ @@ -531,10 +532,12 @@ void gtk_status_stack_update_current_instruction(GtkStatusStack *stack, const GL /* Bascule vers une zone courante nouvelle ? */ - range = g_arch_instruction_get_range(instr); addr = get_mrange_addr(range); + size = get_mrange_length(range); - if (cmp_mrange(&info->current, range) == 0) + if (cmp_mrange(&info->current, range) == 0 + && info->size == size + && info->encoding == encoding) goto gssuci_useless; /* Réinitialisation */ @@ -566,9 +569,8 @@ void gtk_status_stack_update_current_instruction(GtkStatusStack *stack, const GL vmpa2_virt_to_string(addr, MDS_UNDEFINED, info->virt, NULL); - info->encoding = g_arch_instruction_get_encoding(instr); - - info->size = get_mrange_length(range); + info->encoding = encoding; + info->size = size; /* Symbole concerné */ @@ -597,7 +599,7 @@ void gtk_status_stack_update_current_instruction(GtkStatusStack *stack, const GL info->reset = false; - gtk_status_stack_show_current_instruction(stack); + gtk_status_stack_show_current_location(stack); gssuci_useless: @@ -618,7 +620,7 @@ void gtk_status_stack_update_current_instruction(GtkStatusStack *stack, const GL * * ******************************************************************************/ -void gtk_status_stack_reset_current_instruction(GtkStatusStack *stack) +void gtk_status_stack_reset_current_location(GtkStatusStack *stack) { assembly_info *info; /* Informations à constituer */ @@ -626,7 +628,7 @@ void gtk_status_stack_reset_current_instruction(GtkStatusStack *stack) reset_assembly_info(info); - gtk_status_stack_show_current_instruction(stack); + gtk_status_stack_show_current_location(stack); } @@ -643,7 +645,7 @@ void gtk_status_stack_reset_current_instruction(GtkStatusStack *stack) * * ******************************************************************************/ -static gboolean gtk_status_stack_show_current_instruction(GtkStatusStack *stack) +static gboolean gtk_status_stack_show_current_location(GtkStatusStack *stack) { GObject *ref; /* Espace de référencements */ assembly_info *info; /* Informations à consulter */ @@ -651,7 +653,7 @@ static gboolean gtk_status_stack_show_current_instruction(GtkStatusStack *stack) char raw_pos[6 + VMPA_MAX_LEN + 1]; /* Formatage final en direct */ char *content; /* Contenu dynamique */ - stack->def_source = (GSourceFunc)gtk_status_stack_show_current_instruction; + stack->def_source = (GSourceFunc)gtk_status_stack_show_current_location; gtk_stack_set_visible_child_name(stack->main, "asm_info"); @@ -697,7 +699,7 @@ static gboolean gtk_status_stack_show_current_instruction(GtkStatusStack *stack) /* Seconde partie : architecture */ - if (info->reset) + if (info->reset || info->encoding == NULL || info->size == VMPA_NO_PHYSICAL) { label = GTK_LABEL(g_object_get_data(ref, "arch")); gtk_label_set_text(label, NULL); diff --git a/src/gtkext/gtkstatusstack.h b/src/gtkext/gtkstatusstack.h index bcb9477..21c9b58 100644 --- a/src/gtkext/gtkstatusstack.h +++ b/src/gtkext/gtkstatusstack.h @@ -62,10 +62,10 @@ GtkWidget *gtk_status_stack_new(void); /* Actualise les informations liées une position d'assemblage. */ -void gtk_status_stack_update_current_instruction(GtkStatusStack *, const GLoadedBinary *, const GArchInstruction *); +void gtk_status_stack_update_current_location(GtkStatusStack *, const GLoadedBinary *, const mrange_t *, const char *); /* Réinitialise les informations associées une position. */ -void gtk_status_stack_reset_current_instruction(GtkStatusStack *); +void gtk_status_stack_reset_current_location(GtkStatusStack *); diff --git a/src/gtkext/hexdisplay.c b/src/gtkext/hexdisplay.c new file mode 100644 index 0000000..8cb4430 --- /dev/null +++ b/src/gtkext/hexdisplay.c @@ -0,0 +1,332 @@ + +/* Chrysalide - Outil d'analyse de fichiers binaires + * hexdisplay.c - affichage d'un contenu binaire sous forme hexadécimale + * + * Copyright (C) 2016 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 Foobar. If not, see <http://www.gnu.org/licenses/>. + */ + + +#include "hexdisplay.h" + + +#include "gtkbufferdisplay-int.h" +#include "../format/format.h" +#include "../glibext/generators/hex.h" + + + +/* Composant d'affichage de contenu sous forme hexadécimale (instance) */ +struct _GtkHexDisplay +{ + GtkBufferDisplay parent; /* A laisser en premier */ + + GBufferCache *cache; /* Cache pour l'affichage */ + GHexGenerator *generator; /* Générateur à la volée */ + +}; + +/* Composant d'affichage de contenu sous forme hexadécimale (classe) */ +struct _GtkHexDisplayClass +{ + GtkBufferDisplayClass parent; /* A laisser en premier */ + +}; + + +/* Procède à l'initialisation des afficheurs sous forme hexa. */ +static void gtk_hex_display_class_init(GtkHexDisplayClass *); + +/* Procède à l'initialisation de l'afficheur sous forme hexa. */ +static void gtk_hex_display_init(GtkHexDisplay *); + +/* Supprime toutes les références externes. */ +static void g_hex_display_dispose(GtkHexDisplay *); + +/* Procède à la libération totale de la mémoire. */ +static void g_hex_display_finalize(GtkHexDisplay *); + +/* S'adapte à la surface concédée par le composant parent. */ +static void gtk_hex_display_size_allocate(GtkWidget *, GtkAllocation *); + +/* Indique les dimensions de travail du composant d'affichage. */ +static void gtk_hex_display_compute_requested_size(GtkHexDisplay *, gint *, gint *); + +/* Adapte le cache de lignes hexadécimales à la taille courante. */ +static void gtk_hex_display_populate_cache(GtkHexDisplay *); + + + +/* Détermine le type du composant d'affichage sous forme hexadécimale. */ +G_DEFINE_TYPE(GtkHexDisplay, gtk_hex_display, GTK_TYPE_BUFFER_DISPLAY) + + +/****************************************************************************** +* * +* Paramètres : class = classe GTK à initialiser. * +* * +* Description : Procède à l'initialisation des afficheurs sous forme hexa. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void gtk_hex_display_class_init(GtkHexDisplayClass *class) +{ + GObjectClass *object; /* Autre version de la classe */ + GtkWidgetClass *widget_class; /* Classe de haut niveau */ + GtkDisplayPanelClass *panel_class; /* Classe parente */ + + object = G_OBJECT_CLASS(class); + + object->dispose = (GObjectFinalizeFunc/* ! */)g_hex_display_dispose; + object->finalize = (GObjectFinalizeFunc)g_hex_display_finalize; + + widget_class = GTK_WIDGET_CLASS(class); + + widget_class->size_allocate = gtk_hex_display_size_allocate; + + panel_class = GTK_DISPLAY_PANEL_CLASS(class); + + panel_class->compute_size = (compute_requested_size_fc)gtk_hex_display_compute_requested_size; + +} + + +/****************************************************************************** +* * +* Paramètres : view = composant GTK à initialiser. * +* * +* Description : Procède à l'initialisation de l'afficheur sous forme hexa. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void gtk_hex_display_init(GtkHexDisplay *view) +{ + +} + + +/****************************************************************************** +* * +* Paramètres : display = instance d'objet GLib à traiter. * +* * +* Description : Supprime toutes les références externes. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void g_hex_display_dispose(GtkHexDisplay *display) +{ + g_clear_object(&display->cache); + + g_clear_object(&display->generator); + + G_OBJECT_CLASS(gtk_hex_display_parent_class)->dispose(G_OBJECT(display)); + +} + + +/****************************************************************************** +* * +* Paramètres : display = instance d'objet GLib à traiter. * +* * +* Description : Procède à la libération totale de la mémoire. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void g_hex_display_finalize(GtkHexDisplay *display) +{ + G_OBJECT_CLASS(gtk_hex_display_parent_class)->finalize(G_OBJECT(display)); + +} + + +/****************************************************************************** +* * +* Paramètres : content = contenu brut à représenter. * +* * +* Description : Crée un nouveau composant pour l'affichage sous forme hexa. * +* * +* Retour : Composant GTK créé. * +* * +* Remarques : - * +* * +******************************************************************************/ + +GtkWidget *gtk_hex_display_new(GBinContent *content) +{ + GtkHexDisplay *result; /* Composant à retourner */ + GBufferView *view; /* Vue pointée sur un tampon */ + + result = g_object_new(GTK_TYPE_HEX_DISPLAY, NULL); + + result->cache = g_buffer_cache_new(content); + g_object_ref_sink(G_OBJECT(result->cache)); + + result->generator = g_hex_generator_new(content); + + gtk_hex_display_populate_cache(result); + + view = g_buffer_view_new(result->cache, NULL); + + GTK_BUFFER_DISPLAY(result)->view = view; + + return GTK_WIDGET(result); + +} + + +/****************************************************************************** +* * +* Paramètres : widget = composant GTK à mettre à jour. * +* allocation = étendue accordée à la vue. * +* * +* Description : S'adapte à la surface concédée par le composant parent. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void gtk_hex_display_size_allocate(GtkWidget *widget, GtkAllocation *allocation) +{ + GtkHexDisplay *display; /* Autre version du composant */ + GBufferCache *cache; /* Contenu représenté */ + gint text_pos; /* Abscisse minimale du texte */ + bool show_pos; /* Affichage des positions ? */ + bool changed; /* Note toute variation */ + + display = GTK_HEX_DISPLAY(widget); + + cache = g_buffer_view_get_cache(GTK_BUFFER_DISPLAY(display)->view); + + text_pos = g_buffer_cache_get_text_position(cache); + + g_object_unref(G_OBJECT(cache)); + + show_pos = g_display_options_get(GTK_DISPLAY_PANEL(widget)->options, 0); + + changed = g_hex_generator_auto_fit(display->generator, text_pos, show_pos, allocation->width); + + if (changed) + gtk_hex_display_populate_cache(display); + + /** + * On fait appel au parent en dernier pour bénéficier des besoins + * en espace actualisés avec les nouvelles dispositions. + */ + + GTK_WIDGET_CLASS(gtk_hex_display_parent_class)->size_allocate(widget, allocation); + +} + + +/****************************************************************************** +* * +* Paramètres : display = composant GTK à consulter. * +* width = largeur requise à renseigner ou NULL. [OUT] * +* height = hauteur requise à renseigner ou NULL. [OUT] * +* * +* Description : Indique les dimensions de travail du composant d'affichage. * +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void gtk_hex_display_compute_requested_size(GtkHexDisplay *display, gint *width, gint *height) +{ + GtkDisplayPanel *pdisplay; /* Version parente */ + + pdisplay = GTK_DISPLAY_PANEL(display); + + GTK_DISPLAY_PANEL_CLASS(gtk_hex_display_parent_class)->compute_size(pdisplay, width, height); + + if (width != NULL && *width != 0) + *width = 1; + +} + + +/****************************************************************************** +* * +* Paramètres : display = composant GTK à mettre à jour. * +* * +* Description : Adapte le cache de lignes hexadécimales à la taille courante.* +* * +* Retour : - * +* * +* Remarques : - * +* * +******************************************************************************/ + +static void gtk_hex_display_populate_cache(GtkHexDisplay *display) +{ + GBinContent *content; /* Contenu binaire affiché */ + phys_t full; /* Taille totale à représenter */ + phys_t line; /* Taille représentée par ligne*/ + size_t needed; /* Nombre de lignes nécessaires*/ + size_t count; /* Nombre actuel de lignes */ + + /* Détermination du besoin */ + + content = g_hex_generator_get_content(display->generator); + + full = g_binary_content_compute_size(content); + + g_object_unref(G_OBJECT(content)); + + line = g_hex_generator_get_bytes_per_line(display->generator); + + needed = full / line; + + if (full % line > 0) + needed++; + + /* Adaptation du tampon interne */ + + count = g_buffer_cache_count_lines(display->cache); + + if (needed < count) + g_buffer_cache_truncate(display->cache, needed); + + else if (needed > count) + { + g_object_ref(G_OBJECT(display->generator)); + g_buffer_cache_extend_with(display->cache, needed, G_LINE_GENERATOR(display->generator)); + } + + if (needed != count) + gtk_widget_queue_resize(GTK_WIDGET(display)); + +} diff --git a/src/gtkext/hexdisplay.h b/src/gtkext/hexdisplay.h new file mode 100644 index 0000000..c806555 --- /dev/null +++ b/src/gtkext/hexdisplay.h @@ -0,0 +1,59 @@ + +/* Chrysalide - Outil d'analyse de fichiers binaires + * hexdisplay.h - prototypes pour l'affichage d'un contenu binaire sous forme hexadécimale + * + * Copyright (C) 2016 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 Foobar. If not, see <http://www.gnu.org/licenses/>. + */ + + +#ifndef _GTKEXT_HEXDISPLAY_H +#define _GTKEXT_HEXDISPLAY_H + + +#include <glib-object.h> +#include <gtk/gtk.h> + + +#include "../analysis/content.h" + + + +#define GTK_TYPE_HEX_DISPLAY (gtk_hex_display_get_type()) +#define GTK_HEX_DISPLAY(obj) (G_TYPE_CHECK_INSTANCE_CAST((obj), GTK_TYPE_HEX_DISPLAY, GtkHexDisplay)) +#define GTK_HEX_DISPLAY_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST((klass), GTK_TYPE_HEX_DISPLAY, GtkHexDisplayClass)) +#define GTK_IS_HEX_DISPLAY(obj) (G_TYPE_CHECK_INSTANCE_TYPE((obj), GTK_TYPE_HEX_DISPLAY)) +#define GTK_IS_HEX_DISPLAY_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE((klass), GTK_TYPE_HEX_DISPLAY)) +#define GTK_HEX_DISPLAY_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS((obj), GTK_TYPE_HEX_DISPLAY, GtkHexDisplayClass)) + + +/* Composant d'affichage de contenu sous forme hexadécimale (instance) */ +typedef struct _GtkHexDisplay GtkHexDisplay; + +/* Composant d'affichage de contenu sous forme hexadécimale (classe) */ +typedef struct _GtkHexDisplayClass GtkHexDisplayClass; + + +/* Détermine le type du composant d'affichage sous forme hexadécimale. */ +GType gtk_hex_display_get_type(void); + +/* Crée un nouveau composant pour l'affichage sous forme hexa. */ +GtkWidget *gtk_hex_display_new(GBinContent *); + + + +#endif /* _GTKEXT_HEXDISPLAY_H */ diff --git a/src/gui/menus/edition.c b/src/gui/menus/edition.c index d3381bf..2cfcf85 100644 --- a/src/gui/menus/edition.c +++ b/src/gui/menus/edition.c @@ -44,6 +44,7 @@ #include "../../gtkext/gtkblockdisplay.h" #include "../../gtkext/gtkdisplaypanel.h" #include "../../gtkext/gtkgraphdisplay.h" +#include "../../gtkext/hexdisplay.h" @@ -353,7 +354,7 @@ void update_access_for_cursor_in_menu_edition(GLoadedPanel *panel, const GLineCu if (G_IS_BINARY_CURSOR(cursor)) { - assert(GTK_IS_BLOCK_DISPLAY(panel) || GTK_IS_GRAPH_DISPLAY(panel)); + assert(GTK_IS_HEX_DISPLAY(panel) || GTK_IS_BLOCK_DISPLAY(panel) || GTK_IS_GRAPH_DISPLAY(panel)); if (g_line_cursor_is_valid(cursor)) creator = gtk_display_panel_get_active_object(GTK_DISPLAY_PANEL(panel)); @@ -433,7 +434,7 @@ static void mcb_edition_goto(GtkMenuItem *menuitem, GMenuBar *bar) addr = get_address_from_goto_dialog(dialog); panel = get_current_view(); - assert(GTK_IS_BLOCK_DISPLAY(panel) || GTK_IS_GRAPH_DISPLAY(panel)); + assert(GTK_IS_HEX_DISPLAY(panel) || GTK_IS_BLOCK_DISPLAY(panel) || GTK_IS_GRAPH_DISPLAY(panel)); binary = G_LOADED_BINARY(g_loaded_panel_get_content(panel)); g_loaded_binary_remember_new_goto(binary, addr); diff --git a/themes/Adwaita/display.css b/themes/Adwaita/display.css index f5bd72c..977490d 100644 --- a/themes/Adwaita/display.css +++ b/themes/Adwaita/display.css @@ -3,8 +3,6 @@ color: #000; - font-style: normal; - font-weight: normal; } @@ -12,8 +10,6 @@ color: #37b05b; - font-style: normal; - font-weight: normal; } @@ -22,7 +18,6 @@ color: #828282; font-style: italic; - font-weight: normal; } @@ -30,62 +25,42 @@ color: darker(#7c7c7b); - font-style: normal; - font-weight: normal; - } .segment-phys-addr { color: #7c7c7b; - font-style: normal; - font-weight: normal; - } .segment-virt-addr-padding { color: darker(#7c7c7b); - font-style: normal; - font-weight: normal; - } .segment-virt-addr { color: #7c7c7b; - font-style: normal; - font-weight: normal; - } .segment-raw-code { color: #000; - font-style: normal; - font-weight: normal; - } .segment-label { color: #ffd200; - font-style: normal; - font-weight: normal; - } .segment-instruction { color: #bebebe; - font-style: normal; - font-weight: normal; } @@ -93,8 +68,6 @@ color: #a020f0; - font-style: normal; - font-weight: normal; } @@ -102,8 +75,6 @@ color: #4141d1; - font-style: normal; - font-weight: normal; } @@ -111,7 +82,6 @@ color: #000; - font-style: normal; font-weight: bold; } @@ -120,7 +90,6 @@ color: #000; - font-style: normal; font-weight: bold; } @@ -129,7 +98,6 @@ color: #000; - font-style: normal; font-weight: bold; } @@ -138,8 +106,6 @@ color: #000; - font-style: normal; - font-weight: normal; } @@ -147,8 +113,6 @@ color: #c80a0a; - font-style: normal; - font-weight: normal; } @@ -156,8 +120,6 @@ color: #000; - font-style: normal; - font-weight: normal; } @@ -165,8 +127,6 @@ color: #cc7e00; - font-style: normal; - font-weight: normal; } @@ -174,8 +134,6 @@ color: #000; - font-style: normal; - font-weight: normal; } @@ -183,8 +141,6 @@ color: #000; - font-style: normal; - font-weight: normal; } @@ -192,7 +148,6 @@ color: #ff0000; - font-style: normal; font-weight: bold; } diff --git a/themes/Makefile.am b/themes/Makefile.am index 8010143..d75b446 100644 --- a/themes/Makefile.am +++ b/themes/Makefile.am @@ -1,2 +1,4 @@ -SUBDIRS = Adwaita +SUBDIRS = \ + Adwaita \ + Solarized diff --git a/themes/Solarized/Makefile.am b/themes/Solarized/Makefile.am new file mode 100644 index 0000000..37631eb --- /dev/null +++ b/themes/Solarized/Makefile.am @@ -0,0 +1,18 @@ + +GTK3_CSS = \ + palette.css \ + widgets.css \ + widgets-dark.css \ + segments.css \ + segments-dark.css + +solarized.ctm: gresource.xml $(GTK3_CSS) + glib-compile-resources --target=$@ --sourcedir=$(srcdir) gresource.xml + +css_DATA = solarized.ctm + +cssdir = $(themesdir) + +EXTRA_DIST = $(css_DATA) gresource.xml + +CLEANFILES = $(css_DATA) diff --git a/themes/Solarized/definition.xml b/themes/Solarized/definition.xml new file mode 100644 index 0000000..5fc3e83 --- /dev/null +++ b/themes/Solarized/definition.xml @@ -0,0 +1,22 @@ +<?xml version="1.0" encoding="UTF-8"?> +<ChrysalideTheme> + <identity> + <name>Solarized</name> + <author>Cyrille Bagard</author> + <desc>Precision colors for Chrysalide based on https://ethanschoonover.com/solarized/</desc> + </identity> + <resources> + <common> + <path>palette.css</path> + <path>segments.css</path> + </common> + <light> + <path>widgets-light.css</path> + <path>segments-light.css</path> + </light> + <dark> + <path>widgets-dark.css</path> + <path>segments-dark.css</path> + </dark> + </resources> +</ChrysalideTheme> diff --git a/themes/Solarized/gresource.xml b/themes/Solarized/gresource.xml new file mode 100644 index 0000000..a12dcba --- /dev/null +++ b/themes/Solarized/gresource.xml @@ -0,0 +1,12 @@ +<?xml version="1.0" encoding="UTF-8"?> +<gresources> + <gresource prefix="/org/chrysalide/gui/themes/Solarized"> + <file compressed="true">definition.xml</file> + <file compressed="true">palette.css</file> + <file compressed="true">segments.css</file> + <file compressed="true">widgets-light.css</file> + <file compressed="true">segments-light.css</file> + <file compressed="true">widgets-dark.css</file> + <file compressed="true">segments-dark.css</file> + </gresource> +</gresources> diff --git a/themes/Solarized/palette.css b/themes/Solarized/palette.css new file mode 100644 index 0000000..d111a31 --- /dev/null +++ b/themes/Solarized/palette.css @@ -0,0 +1,17 @@ + +@define-color base03 #002b36; +@define-color base02 #073642; +@define-color base01 #586e75; +@define-color base00 #657b83; +@define-color base0 #839496; +@define-color base1 #93a1a1; +@define-color base2 #eee8d5; +@define-color base3 #fdf6e3; +@define-color yellow #b58900; +@define-color orange #cb4b16; +@define-color red #dc322f; +@define-color magenta #d33682; +@define-color violet #6c71c4; +@define-color blue #268bd2; +@define-color cyan #2aa198; +@define-color green #859900; diff --git a/themes/Solarized/segments-dark.css b/themes/Solarized/segments-dark.css new file mode 100644 index 0000000..3c9a6fd --- /dev/null +++ b/themes/Solarized/segments-dark.css @@ -0,0 +1,61 @@ + +/* Hex display */ + +.segment-raw { + + color: @base1; + +} + +.segment-raw-full { + + color: @base01; + +} + +.segment-raw-null { + + color: @base02; + +} + +.segment-printable { + + color: @base1; + +} + +.segment-not-printable { + + color: @base02; + +} + +/* Assembly views */ + +.segment-phys-addr-padding, +.segment-virt-addr-padding { + + color: @base02; + +} + +.segment-phys-addr, +.segment-virt-addr { + + color: @base01; + +} + +.segment-comment { + + color: @base1; + +} + +.segment-keyword, +.segment-instruction { + + color: @base2; + +} diff --git a/themes/Solarized/segments-light.css b/themes/Solarized/segments-light.css new file mode 100644 index 0000000..74255e4 --- /dev/null +++ b/themes/Solarized/segments-light.css @@ -0,0 +1,61 @@ + +/* Hex display */ + +.segment-raw { + + color: @base01; + +} + +.segment-raw-full { + + color: @base1; + +} + +.segment-raw-null { + + color: @base2; + +} + +.segment-printable { + + color: @base01; + +} + +.segment-not-printable { + + color: @base2; + +} + +/* Assembly views */ + +.segment-phys-addr-padding, +.segment-virt-addr-padding { + + color: @base2; + +} + +.segment-phys-addr, +.segment-virt-addr { + + color: @base1; + +} + +.segment-comment { + + color: @base01; + +} + +.segment-keyword, +.segment-instruction { + + color: @base02; + +} diff --git a/themes/Solarized/segments.css b/themes/Solarized/segments.css new file mode 100644 index 0000000..0391466 --- /dev/null +++ b/themes/Solarized/segments.css @@ -0,0 +1,48 @@ + +/* Assembly views */ + +.segment-label { + + color: @blue; + +} + +.segment-register { + + color: @violet; + +} + +.segment-immediate { + + color: @orange; + +} + +.segment-string { + + color: @yellow; + +} + +.segment-keyword { + + font-style: italic; + +} + +.segment-punct { + + color: @cyan; + +} + +.segment-hooks, +.segment-ltgt, +.segment-signs { + + color: @cyan; + + font-weight: bold; + +} diff --git a/themes/Solarized/widgets-dark.css b/themes/Solarized/widgets-dark.css new file mode 100644 index 0000000..5265d56 --- /dev/null +++ b/themes/Solarized/widgets-dark.css @@ -0,0 +1,6 @@ + +.graph-block-background { + + background-color: @base03; + +} diff --git a/themes/Solarized/widgets-light.css b/themes/Solarized/widgets-light.css new file mode 100644 index 0000000..b55cae2 --- /dev/null +++ b/themes/Solarized/widgets-light.css @@ -0,0 +1,6 @@ + +.graph-block-background { + + background-color: @base3; + +} -- cgit v0.11.2-87-g4458