summaryrefslogtreecommitdiff
path: root/src/analysis/line.c
diff options
context:
space:
mode:
authorCyrille Bagard <nocbos@gmail.com>2009-03-04 01:38:27 (GMT)
committerCyrille Bagard <nocbos@gmail.com>2009-03-04 01:38:27 (GMT)
commit674739bedc853681be5a2657a7a4f497d6e82c9b (patch)
tree10e65041c0607d4e5a20c0a7a77fe75e1bbd7737 /src/analysis/line.c
parentbbd04bf5f23e7dca1917247cb92824805b22a2a4 (diff)
Added a line type for binary code display.
git-svn-id: svn://svn.gna.org/svn/chrysalide/trunk@51 abbe820e-26c8-41b2-8c08-b7b2b41f8b0a
Diffstat (limited to 'src/analysis/line.c')
-rw-r--r--src/analysis/line.c292
1 files changed, 288 insertions, 4 deletions
diff --git a/src/analysis/line.c b/src/analysis/line.c
index 7431e94..052adea 100644
--- a/src/analysis/line.c
+++ b/src/analysis/line.c
@@ -27,6 +27,7 @@
#include <malloc.h>
#include <stdio.h>
#include <string.h>
+#include <sys/param.h>
#include "../common/dllist.h"
@@ -38,6 +39,17 @@ extern GtkWidget *mywid;
+
+
+
+/* Méthode de mise à jour du nombre d'octets maximal par instruction. */
+typedef void (* get_bin_len_fc) (rendering_line *, off_t *);
+
+/* Méthode de mise à jour d'une ligne de représentation. */
+typedef void (* refresh_markup_fc) (rendering_line *);
+
+
+
/* Ligne de représentation générique */
struct _rendering_line
{
@@ -47,6 +59,11 @@ struct _rendering_line
PangoLayout *layout; /* Moteur de rendu du code/txt */
+ get_bin_len_fc get_bin_len; /* Nbre d'octets représentés */
+ off_t max_bin_len; /* Nombre global maximal */
+
+ refresh_markup_fc refresh_markup; /* Reconstruit la représentat° */
+
};
@@ -77,11 +94,29 @@ void refresh_prologue_markup(prologue_line *);
+/* ------------------------ LIGNE DE CODE EN LANGAGE MACHINE ------------------------ */
+
+/* Ligne de représentation de prologue */
+typedef struct _code_line
+{
+ rendering_line basic; /* A laisser en premier */
+
+ asm_instr *instr; /* Instruction représentée */
+ const disass_options *options; /* Options de représentation */
+
+} code_line;
+/* Taille max d'une traduction */
+#define CODE_BUFFER_LEN 128
+/* Met à jour la nombre d'octets maximale par instruction. */
+void get_code_binary_len(code_line *, off_t *);
+
+/* Met à jour la ligne de représentation de code. */
+void refresh_code_markup(code_line *);
@@ -106,6 +141,8 @@ void init_rendering_line(rendering_line *line)
line->layout = gtk_widget_create_pango_layout(mywid, NULL);
+ line->get_bin_len = NULL;
+ line->refresh_markup = NULL;
@@ -137,6 +174,78 @@ void add_line_to_rendering_lines(rendering_line **lines, rendering_line *line)
+
+
+/******************************************************************************
+* *
+* Paramètres : line = ligne de représentation à actualiser. *
+* blen = longueur maximale à mettre à jour. [OUT] *
+* *
+* Description : Met à jour le nombre d'octets maximal par instruction. *
+* *
+* Retour : - *
+* *
+* Remarques : - *
+* *
+******************************************************************************/
+
+void get_rendering_line_binary_len(rendering_line *line, off_t *blen)
+{
+ if (line->get_bin_len != NULL)
+ line->get_bin_len(line, blen);
+
+}
+
+
+/******************************************************************************
+* *
+* Paramètres : line = ligne de représentation à actualiser. *
+* blen = longueur maximale à prendre en compte. *
+* *
+* Description : Prend en compte le nombre d'octets maximal par instruction. *
+* *
+* Retour : - *
+* *
+* Remarques : - *
+* *
+******************************************************************************/
+
+void set_rendering_line_max_binary_len(rendering_line *line, off_t blen)
+{
+ line->max_bin_len = blen * 2 + (blen - 1);
+
+ line->refresh_markup(line);
+
+}
+
+
+/******************************************************************************
+* *
+* Paramètres : line = adresse de la structure à représenter. *
+* width = largeur maximale des lignes à compléter. *
+* height = hauteur maximale des lignes à compléter. *
+* *
+* Description : Fournit les dimensions d'une ligne par rapport à d'autres. *
+* *
+* Retour : - *
+* *
+* Remarques : - *
+* *
+******************************************************************************/
+
+void get_rendering_line_size(rendering_line *line, int *width, int *height)
+{
+ int w; /* Largeur de l'objet actuelle */
+ int h; /* Hauteur de l'objet actuelle */
+
+ pango_layout_get_pixel_size(line->layout, &w, &h);
+
+ *width = MAX(*width, w);
+ *height += h;
+
+}
+
+
/******************************************************************************
* *
* Paramètres : line = adresse de la structure à représenter. *
@@ -170,9 +279,9 @@ void draw_rendering_line(rendering_line *line, GdkDrawable *drawable, GdkGC *gc,
* *
* Paramètres : comment = texte à afficher au final. *
* *
-* Description : Choisit d'afficher le code brut ou non. *
+* Description : Crée une des lignes de description initiales. *
* *
-* Retour : - *
+* Retour : Adresse de la structure mise en place. *
* *
* Remarques : - *
* *
@@ -188,9 +297,9 @@ rendering_line *create_prologue_line(const char *comment)
RENDERING_LINE(result)->type = RLT_PROLOGUE;
- result->comment = strdup(comment);
+ RENDERING_LINE(result)->refresh_markup = (refresh_markup_fc)refresh_prologue_markup;
- refresh_prologue_markup(result);
+ result->comment = strdup(comment);
return RENDERING_LINE(result);
@@ -229,3 +338,178 @@ void refresh_prologue_markup(prologue_line *line)
}
+
+/* ---------------------------------------------------------------------------------- */
+/* LIGNE DE CODE EN LANGAGE MACHINE */
+/* ---------------------------------------------------------------------------------- */
+
+
+/******************************************************************************
+* *
+* Paramètres : instr = instruction à représenter. *
+* options = paramétrage du rendu. *
+* *
+* Description : Crée une ligne de représentation de code binaire. *
+* *
+* Retour : Adresse de la structure mise en place. *
+* *
+* Remarques : - *
+* *
+******************************************************************************/
+
+rendering_line *create_code_line(asm_instr *instr, const disass_options *options)
+{
+ code_line *result; /* Structure à retourner */
+
+ result = (code_line *)calloc(1, sizeof(code_line));
+
+ init_rendering_line(RENDERING_LINE(result));
+
+ RENDERING_LINE(result)->type = RLT_CODE;
+
+ RENDERING_LINE(result)->get_bin_len = (get_bin_len_fc)get_code_binary_len;
+ RENDERING_LINE(result)->refresh_markup = (refresh_markup_fc)refresh_code_markup;
+
+ result->instr = instr;
+ result->options = options;
+
+ return RENDERING_LINE(result);
+
+}
+
+
+/******************************************************************************
+* *
+* Paramètres : line = ligne de représentation à actualiser. *
+* blen = longueur maximale à mettre à jour. [OUT] *
+* *
+* Description : Met à jour le nombre d'octets maximal par instruction. *
+* *
+* Retour : - *
+* *
+* Remarques : - *
+* *
+******************************************************************************/
+
+void get_code_binary_len(code_line *line, off_t *blen)
+{
+ off_t len; /* Taille propre à la ligne */
+
+ get_asm_instr_offset_and_length(line->instr, NULL, &len);
+
+ *blen = MAX(*blen, len);
+
+}
+
+
+/******************************************************************************
+* *
+* Paramètres : line = ligne de représentation à actualiser. *
+* *
+* Description : Met à jour la ligne de représentation de code. *
+* *
+* Retour : - *
+* *
+* Remarques : - *
+* *
+******************************************************************************/
+
+void refresh_code_markup(code_line *line)
+{
+ size_t len; /* Taille du contenu */
+ char *content; /* Contenu réellement imprimé */
+ off_t bin_offset; /* Début de l'instruction */
+ off_t bin_len; /* Taille d'instruction */
+ char buffer[CODE_BUFFER_LEN]; /* Zone tampon à utiliser */
+ const uint8_t *exe_content; /* Contenu binaire global */
+ char *bin_code; /* Tampon du code binaire */
+ off_t k; /* Boucle de parcours #2 */
+ off_t j; /* Boucle de parcours #1 */
+
+ len = strlen("<tt>") + 1;
+ content = (char *)calloc(len, sizeof(char));
+ strcpy(content, "<tt>");
+
+ if (line->options->show_address || line->options->show_code)
+ get_asm_instr_offset_and_length(line->instr, &bin_offset, &bin_len);
+
+ /* Eventuelle adresse virtuelle */
+
+ if (line->options->show_address)
+ {
+ switch (ADM_32BITS /* FIXME */)
+ {
+ case ADM_32BITS:
+ snprintf(buffer, CODE_BUFFER_LEN,
+ "<span foreground='#333333'>0x%08llx</span>",
+ bin_offset);
+ break;
+
+ case ADM_64BITS:
+ snprintf(buffer, CODE_BUFFER_LEN,
+ "<span foreground='#333333'>0x%16llx</span>",
+ bin_offset);
+ break;
+
+ }
+
+ len += strlen(buffer);
+ content = (char *)realloc(content, len * sizeof(char));
+ strcat(content, buffer);
+
+ }
+
+ /* Eventuel code brut */
+
+ if (line->options->show_code)
+ {
+ exe_content = get_exe_content(line->options->format, NULL);
+
+ bin_code = (char *)calloc(RENDERING_LINE(line)->max_bin_len + 1, sizeof(char));
+
+ k = 0;
+
+ for (j = 0; j < bin_len; j++)
+ {
+ if ((j + 1) < bin_len)
+ k += snprintf(&bin_code[j * (2 + 1)], 4, "%02hhx ", exe_content[bin_offset + j]);
+ else
+ k += snprintf(&bin_code[j * (2 + 1)], 3, "%02hhx", exe_content[bin_offset + j]);
+ }
+
+ for (; k < RENDERING_LINE(line)->max_bin_len; k++)
+ snprintf(&bin_code[k], 2, " ");
+
+ if (line->options->show_address) len += strlen("\t");
+ len += strlen(bin_code);
+ content = (char *)realloc(content, len * sizeof(char));
+ if (line->options->show_address) strcat(content, "\t");
+ strcat(content, bin_code);
+
+ free(bin_code);
+
+ }
+
+ /* Instruction proprement dite */
+
+ print_hinstruction(line->options->proc, line->options->format,
+ line->instr, buffer, CODE_BUFFER_LEN, ASX_INTEL/*FIXME*/);
+
+ if (line->options->show_address || line->options->show_code) len += strlen("\t");
+ len += strlen(buffer);
+
+ content = (char *)realloc(content, len * sizeof(char));
+ if (line->options->show_address || line->options->show_code) strcat(content, "\t");
+ strcat(content, buffer);
+
+ /* Finalisation */
+
+ len += strlen("</tt>");
+ content = (char *)realloc(content, len * sizeof(char));
+ strcat(content, "</tt>");
+
+ pango_layout_set_markup(RENDERING_LINE(line)->layout, content, len - 1);
+
+ free(content);
+
+}