/* OpenIDA - Outil d'analyse de fichiers binaires * line_code.c - représentation des lignes de code binaire. * * 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 "line_code.h" #include #include #include "line-int.h" #include "../format/format.h" /* Ligne de représentation de code binaire (instance) */ struct _GCodeLine { GRenderingLine parent; /* Instance parente */ GArchInstruction *instr; /* Instruction représentée */ const GRenderingOptions *options; /* Options de représentation */ }; /* Ligne de représentation de code binaire (classe) */ struct _GCodeLineClass { GRenderingLineClass parent; /* Classe parente */ }; /* Initialise la classe des lignes de code binaire. */ static void g_code_line_class_init(GCodeLineClass *); /* Initialise la classe des lignes de code binaire. */ static void g_code_line_init(GCodeLine *); /* Ajoute du texte simple à un fichier ouvert en écriture. */ static void g_code_line_add_text(GCodeLine *, GRenderingOptions *, MainRendering, FILE *); /* Ajoute à un tampon GLib le contenu de l'instance spécifiée. */ static void g_code_line_to_buffer(GCodeLine *, GBufferLine *, GRenderingOptions *); /* Ajoute à un texte GTK le contenu de la ligne de code. */ static void g_code_line_add_to_gtk_buffer(GCodeLine *, MainRendering, GtkTextBuffer *, GtkTextIter *, size_t [SAR_COUNT]); /* Indique le type définit par la GLib pour la ligne. */ G_DEFINE_TYPE(GCodeLine, g_code_line, G_TYPE_RENDERING_LINE); /****************************************************************************** * * * Paramètres : klass = classe à initialiser. * * * * Description : Initialise la classe des lignes de code binaire. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void g_code_line_class_init(GCodeLineClass *klass) { } /****************************************************************************** * * * Paramètres : line = instance à initialiser. * * * * Description : Initialise la classe des lignes de code binaire. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void g_code_line_init(GCodeLine *line) { GContentExporter *exporter_parent; /* Instance parente #1 */ GRenderingLine *line_parent; /* Instance parente #2 */ exporter_parent = G_CONTENT_EXPORTER(line); exporter_parent->add_text = (add_text_fc)g_code_line_add_text; exporter_parent->export_buffer = (export_buffer_fc)g_code_line_to_buffer; exporter_parent->add_to_gtk_buffer = (add_to_gtk_buffer_fc)g_code_line_add_to_gtk_buffer; line_parent = G_RENDERING_LINE(line); line_parent->type = RLT_CODE; } /****************************************************************************** * * * Paramètres : line = ligne de représentation à représenter. * * options = options de rendu. * * rendering = support effectif final des lignes de code. * * stream = flux ouvert en écriture. * * * * Description : Ajoute du texte simple à un fichier ouvert en écriture. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void g_code_line_add_text(GCodeLine *line, GRenderingOptions *options, MainRendering rendering, FILE *stream) { GContentExporter *exporter; /* Autre vision de la ligne #1 */ GRenderingLine *basic; /* Autre vision de la ligne #2 */ bool show_address; /* Affichage de l'adresse ? */ bool show_code; /* Affichage du code brut ? */ MemoryDataSize msize; /* Taille du bus d'adresses */ char address[VMPA_MAX_SIZE]; /* Adresse au format texte */ size_t len; /* Taille de l'élément inséré */ const bin_t *content; /* Contenu binaire global */ off_t bin_offset; /* Début de l'instruction */ off_t bin_len; /* Taille d'instruction */ char *bin_code; /* Tampon du code binaire */ off_t i; /* Boucle de parcours */ exporter = G_CONTENT_EXPORTER(line); basic = G_RENDERING_LINE(line); show_address = g_rendering_options_has_to_show_address(options, rendering); show_code = g_rendering_options_has_to_show_code(options, rendering); /* Eventuelle adresse virtuelle ou physique */ if (show_address) { msize = g_arch_processor_get_memory_size(g_rendering_options_get_processor(options)); len = vmpa_to_string(G_RENDERING_LINE(line)->offset, msize, address); g_content_exporter_insert_text(exporter, stream, address, len, RTT_NONE); g_content_exporter_insert_text(exporter, stream, "\t", 1, RTT_NONE); } /* Eventuel code brut */ if (show_code) { content = g_binary_format_get_content(G_BIN_FORMAT(g_rendering_options_get_format(options)), NULL); g_arch_instruction_get_location(line->instr, &bin_offset, &bin_len, NULL); bin_code = (char *)calloc(bin_len * 3, sizeof(char)); for (i = 0; i < bin_len; i++) { if ((i + 1) < bin_len) snprintf(&bin_code[i * (2 + 1)], 4, "%02hhx ", content[bin_offset + i]); else snprintf(&bin_code[i * (2 + 1)], 3, "%02hhx", content[bin_offset + i]); } g_content_exporter_insert_text(exporter, stream, bin_code, bin_len * 3 - 1, RTT_RAW_CODE); free(bin_code); g_content_exporter_insert_text(exporter, stream, "\t", 1, RTT_NONE); } /* Instruction proprement dite */ g_content_exporter_add_text(G_CONTENT_EXPORTER(line->instr), options, rendering, stream); /* Commentaire ? */ if (basic->comment != NULL) { g_content_exporter_insert_text(exporter, stream, "\t", 1, RTT_NONE); g_content_exporter_insert_text(exporter, stream, "; ", 2, RTT_COMMENT); g_content_exporter_insert_text(exporter, stream, basic->comment, strlen(basic->comment), RTT_COMMENT); } } /****************************************************************************** * * * Paramètres : line = ligne de représentation à représenter. * * buffer = espace où placer ledit contenu. * * options = options de rendu. * * * * Description : Ajoute à un tampon GLib le contenu de l'instance spécifiée. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void g_code_line_to_buffer(GCodeLine *line, GBufferLine *buffer, GRenderingOptions *options) { GContentExporter *exporter; /* Autre vision de la ligne #1 */ GRenderingLine *basic; /* Autre vision de la ligne #2 */ MemoryDataSize msize; /* Taille du bus d'adresses */ char address[VMPA_MAX_SIZE]; /* Adresse au format texte */ size_t len; /* Taille de l'élément inséré */ const bin_t *content; /* Contenu binaire global */ off_t bin_offset; /* Début de l'instruction */ off_t bin_len; /* Taille d'instruction */ char *bin_code; /* Tampon du code binaire */ off_t i; /* Boucle de parcours */ exporter = G_CONTENT_EXPORTER(line); basic = G_RENDERING_LINE(line); /* Eventuelle adresse virtuelle ou physique */ msize = g_arch_processor_get_memory_size(g_rendering_options_get_processor(line->options)); len = vmpa_to_string(G_RENDERING_LINE(line)->offset, msize, address); g_content_exporter_insert_into_buffer(exporter, buffer, BLC_ADDRESS, address, len, RTT_RAW); /* Eventuel code brut */ content = g_binary_format_get_content(G_BIN_FORMAT(g_rendering_options_get_format(line->options)), NULL); g_arch_instruction_get_location(line->instr, &bin_offset, &bin_len, NULL); bin_code = (char *)calloc(bin_len * 3, sizeof(char)); for (i = 0; i < bin_len; i++) { if ((i + 1) < bin_len) snprintf(&bin_code[i * (2 + 1)], 4, "%02hhx ", content[bin_offset + i]); else snprintf(&bin_code[i * (2 + 1)], 3, "%02hhx", content[bin_offset + i]); } g_content_exporter_insert_into_buffer(exporter, buffer, BLC_BINARY, bin_code, bin_len * 3 - 1, RTT_RAW_CODE); free(bin_code); /* Instruction proprement dite */ g_content_exporter_to_buffer(G_CONTENT_EXPORTER(line->instr), buffer, options); /* Commentaire ? */ if (basic->comment != NULL) { g_content_exporter_insert_into_buffer(exporter, buffer, BLC_COMMENTS, "; ", 2, RTT_COMMENT); g_content_exporter_insert_into_buffer(exporter, buffer, BLC_COMMENTS, basic->comment, strlen(basic->comment), RTT_COMMENT); } } /****************************************************************************** * * * Paramètres : line = ligne de représentation à actualiser. * * rendering = support effectif final des lignes de code. * * buffer = zone de texte à venir compléter. * * iter = point d'insertion du nouveau texte. * * lengths = taille des différentes composantes de la ligne. * * * * Description : Ajoute à un texte GTK le contenu de la ligne de code. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void g_code_line_add_to_gtk_buffer(GCodeLine *line, MainRendering rendering, GtkTextBuffer *buffer, GtkTextIter *iter, size_t lengths[SAR_COUNT]) { GContentExporter *exporter; /* Autre vision de la ligne #1 */ GRenderingLine *basic; /* Autre vision de la ligne #2 */ bool show_address; /* Affichage de l'adresse ? */ bool show_code; /* Affichage du code brut ? */ MemoryDataSize msize; /* Taille du bus d'adresses */ char address[VMPA_MAX_SIZE]; /* Adresse au format texte */ size_t len; /* Taille de l'élément inséré */ const bin_t *content; /* Contenu binaire global */ off_t bin_offset; /* Début de l'instruction */ off_t bin_len; /* Taille d'instruction */ char *bin_code; /* Tampon du code binaire */ off_t i; /* Boucle de parcours */ exporter = G_CONTENT_EXPORTER(line); basic = G_RENDERING_LINE(line); show_address = g_rendering_options_has_to_show_address(line->options, rendering); show_code = g_rendering_options_has_to_show_code(line->options, rendering); /* Eventuelle adresse virtuelle ou physique */ if (show_address) { msize = g_arch_processor_get_memory_size(g_rendering_options_get_processor(line->options)); len = vmpa_to_string(G_RENDERING_LINE(line)->offset, msize, address); lengths[SAR_ADDRESS] = len; g_content_exporter_insert_with_gtk_tag(exporter, buffer, iter, address, len, RTT_NONE); g_content_exporter_insert_with_gtk_tag(exporter, buffer, iter, "\t", 1, RTT_NONE); } /* Eventuel code brut */ if (show_code) { content = g_binary_format_get_content(G_BIN_FORMAT(g_rendering_options_get_format(line->options)), NULL); g_arch_instruction_get_location(line->instr, &bin_offset, &bin_len, NULL); lengths[SAR_CODE] = MAX(lengths[SAR_CODE], bin_len * 3 - 1); bin_code = (char *)calloc(bin_len * 3, sizeof(char)); for (i = 0; i < bin_len; i++) { if ((i + 1) < bin_len) snprintf(&bin_code[i * (2 + 1)], 4, "%02hhx ", content[bin_offset + i]); else snprintf(&bin_code[i * (2 + 1)], 3, "%02hhx", content[bin_offset + i]); } g_content_exporter_insert_with_gtk_tag(exporter, buffer, iter, bin_code, bin_len * 3 - 1, RTT_RAW_CODE); free(bin_code); g_content_exporter_insert_with_gtk_tag(exporter, buffer, iter, "\t", 1, RTT_NONE); } /* Instruction proprement dite */ g_content_exporter_add_arch_to_gtk_buffer(G_CONTENT_EXPORTER(line->instr), g_rendering_options_get_format(line->options), ASX_INTEL/*FIXME*/, buffer, iter); lengths[SAR_INSTRUCTION] = MAX(lengths[SAR_INSTRUCTION], 4 /* FIXME */); /* Commentaire ? */ if (basic->comment != NULL) { g_content_exporter_insert_with_gtk_tag(exporter, buffer, iter, "\t", 1, RTT_NONE); g_content_exporter_insert_with_gtk_tag(exporter, buffer, iter, "; ", 2, RTT_COMMENT); g_content_exporter_insert_with_gtk_tag(exporter, buffer, iter, basic->comment, strlen(basic->comment), RTT_COMMENT); } } /****************************************************************************** * * * Paramètres : offset = emplacement physique ou en mémoire. * * instr = instruction à représenter. * * options = paramétrage du rendu. * * * * Description : Crée une ligne de code binaire. * * * * Retour : Adresse de la structure mise en place. * * * * Remarques : - * * * ******************************************************************************/ GRenderingLine *g_code_line_new(uint64_t offset, GArchInstruction *instr, const GRenderingOptions *options) { GCodeLine *result; /* Structure à retourner */ result = g_object_new(G_TYPE_CODE_LINE, NULL); G_RENDERING_LINE(result)->offset = offset; g_arch_instruction_get_location(instr, NULL, &G_RENDERING_LINE(result)->length, NULL); result->instr = instr; result->options = options; return G_RENDERING_LINE(result); } /****************************************************************************** * * * Paramètres : line = line de rendu à consulter. * * * * Description : Fournit l'instruction associée à la ligne de code binaire. * * * * Retour : Adresse de l'instruction associée. * * * * Remarques : - * * * ******************************************************************************/ GArchInstruction *g_code_line_get_instruction(const GCodeLine *line) { return line->instr; }