diff options
author | Cyrille Bagard <nocbos@gmail.com> | 2019-11-09 13:18:35 (GMT) |
---|---|---|
committer | Cyrille Bagard <nocbos@gmail.com> | 2019-11-09 13:18:35 (GMT) |
commit | c301f7e77eaca632a491b5b4417c8e4b9dce2570 (patch) | |
tree | 501ccdd583e35ccad4f3a0592f2fb1b0870e0d5f /src/glibext/generators | |
parent | 459b345d69532825f21bdcd3e4f92009b0a046dc (diff) |
Introduced the first features of a hexadecimal viewer.
Diffstat (limited to 'src/glibext/generators')
-rw-r--r-- | src/glibext/generators/Makefile.am | 1 | ||||
-rw-r--r-- | src/glibext/generators/hex.c | 617 | ||||
-rw-r--r-- | src/glibext/generators/hex.h | 67 |
3 files changed, 685 insertions, 0 deletions
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 */ |