/* OpenIDA - Outil d'analyse de fichiers binaires
* binary.c - traitement des flots 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 "binary.h"
#include
#include
#include
#include
#include
#include
#include
#include "arch/processor.h"
#include "format/dbg_format.h"
#include "format/exe_format.h"
#include "format/elf/e_elf.h"
#include "format/dwarf/d_dwarf.h"
extern bool find_line_info(const uint8_t *content, off_t *size);
/* Charge en mémoire le contenu d'un fichier à partir d'XML. */
openida_binary *load_binary_file_from_xml(xmlXPathObjectPtr);
/* Charge en mémoire le contenu d'un fichier. */
uint8_t *map_binary_file(const char *, size_t *);
/* Description d'un fichier binaire */
struct _openida_binary
{
char *filename; /* Fichier chargé en mémoire */
};
/******************************************************************************
* *
* Paramètres : filename = nom du fichier à charger. *
* *
* Description : Charge en mémoire le contenu d'un fichier. *
* *
* Retour : Adresse de la représentation ou NULL en cas d'échec. *
* *
* Remarques : - *
* *
******************************************************************************/
openida_binary *load_binary_file(const char *filename)
{
openida_binary *result; /* Adresse à retourner */
result = (openida_binary *)calloc(1, sizeof(openida_binary));
result->filename = strdup(filename);
return result;
}
/******************************************************************************
* *
* Paramètres : xpathObj = point de lecture de tous les éléments. *
* *
* Description : Charge en mémoire le contenu d'un fichier à partir d'XML. *
* *
* Retour : Adresse de la représentation ou NULL en cas d'échec. *
* *
* Remarques : - *
* *
******************************************************************************/
openida_binary *load_binary_file_from_xml(xmlXPathObjectPtr xpathObj)
{
openida_binary *result; /* Adresse à retourner */
int i;
result = (openida_binary *)calloc(1, sizeof(openida_binary));
for (i = 0; i < XPATH_OBJ_NODES_COUNT(xpathObj); i++)
if (xmlStrEqual(NODE_FROM_PATH_OBJ(xpathObj, i)->name, BAD_CAST "Filename"))
result->filename = qck_get_node_text_value(NODE_FROM_PATH_OBJ(xpathObj, i));
return result;
}
/******************************************************************************
* *
* Paramètres : binary = élément binaire à supprimer de la mémoire. *
* *
* Description : Décharge de la mémoire le contenu d'un fichier. *
* *
* Retour : - *
* *
* Remarques : - *
* *
******************************************************************************/
void unload_binary_file(openida_binary *binary)
{
free(binary->filename);
free(binary);
}
/******************************************************************************
* *
* Paramètres : binary = élément binaire à traiter. *
* *
* Description : Fournit une description humaine d'un élément binaire. *
* *
* Retour : Chaîne de caractères humainenement lisible de représentation.*
* *
* Remarques : - *
* *
******************************************************************************/
const char *openida_binary_to_string(const openida_binary *binary)
{
return binary->filename;
}
/******************************************************************************
* *
* Paramètres : xpathCtx = contexte à utiliser pour mener les parcours. *
* base = première partie de l'expression XPath d'accès. *
* index = indice de la élément dans la liste des voisins. *
* *
* Description : Lit un élément binaire depuis un fichier XML. *
* *
* Retour : Représentation mise en place à libérer de la mémoire. *
* *
* Remarques : - *
* *
******************************************************************************/
openida_binary *read_openida_binary_from_xml(xmlXPathContextPtr xpathCtx, const char *base, unsigned int index)
{
openida_binary *result; /* Représentation à retourner */
size_t expr_len; /* Taille d'une expression */
char *expr; /* Chemin XPath reconstitué */
xmlXPathObjectPtr xpathObj; /* Cible d'une recherche */
char *value; /* Type d'élément rencontré */
size_t sub_expr_len; /* Taille d'une expression #2 */
char *sub_expr; /* Chemin XPath reconstitué #2 */
int i; /* Boucle de parcours */
result = NULL;
/* S'occupe en premier lieu du niveau courant */
expr_len = strlen(base) + strlen("/*[position()=") + strlen("4294967295") /* UINT_MAX */ + strlen("]") + 1;
expr = (char *)calloc(expr_len, sizeof(char));
snprintf(expr, expr_len, "%s/*[position()=%u]", base, index);
xpathObj = get_node_xpath_object(xpathCtx, expr);
value = qck_get_node_prop_value(NODE_FROM_PATH_OBJ(xpathObj, 0), "type");
xmlXPathFreeObject(xpathObj);
if (value == NULL) goto robfx_err1;
/* Raffinement au second passage */
sub_expr_len = expr_len + strlen("/*");
sub_expr = (char *)calloc(sub_expr_len, sizeof(char));
snprintf(sub_expr, sub_expr_len, "%s/*", expr);
xpathObj = get_node_xpath_object(xpathCtx, sub_expr);
if (strcmp(value, "file") == 0) result = load_binary_file_from_xml(xpathObj);
xmlXPathFreeObject(xpathObj);
free(sub_expr);
robfx_err1:
free(expr);
return result;
}
/******************************************************************************
* *
* Paramètres : binary = élément binaire à traiter. *
* writer = rédacteur dédié à l'écriture. *
* *
* Description : Ecrit une sauvegarde du binaire dans un fichier XML. *
* *
* Retour : true si l'opération a bien tourné, false sinon. *
* *
* Remarques : - *
* *
******************************************************************************/
bool write_openida_binary_to_xml(const openida_binary *binary, xmlTextWriterPtr writer)
{
bool result; /* Bilan à faire remonter */
result = open_xml_element(writer, "Binary");
result &= write_xml_attribute(writer, "type", "file");
result &= write_xml_element_with_content(writer, "Filename", "%s", binary->filename);
result &= close_xml_element(writer);
return result;
}
/******************************************************************************
* *
* Paramètres : filename = nom du fichier à charger. *
* length = taille des données mises en mémoire. [OUT] *
* *
* Description : Charge en mémoire le contenu d'un fichier. *
* *
* Retour : Adresse du contenu binaire ou NULL en cas d'échec. *
* *
* Remarques : - *
* *
******************************************************************************/
uint8_t *map_binary_file(const char *filename, size_t *length)
{
uint8_t *result; /* Données à retourner */
int fd; /* Fichier ouvert en lecture */
struct stat info; /* Informations sur le fichier */
int ret; /* Bilan d'un appel */
fd = open(filename, 0, O_RDONLY);
if (fd == -1)
{
perror("open()");
return NULL;
}
ret = fstat(fd, &info);
if (ret == -1)
{
perror("fstat()");
close(fd);
return NULL;
}
*length = info.st_size;
result = (uint8_t *)mmap(NULL, *length, PROT_READ, MAP_PRIVATE, fd, 0);
if (result == MAP_FAILED)
{
perror("mmap()");
result = NULL;
}
ret = close(fd);
if (ret == -1)
perror("close()");
return result;
}
void fill_snippet(GtkSnippet *snippet, GtkWidget *panel)
{
off_t length;
uint8_t *bin_data;
int ret;
exe_format *format;
dbg_format *dformat;
asm_processor *proc;
asm_instr *instr;
bin_part **parts;
size_t parts_count;
char **comments;
uint64_t *offsets;
size_t comments_count;
code_line_info **comments_list;
code_line_info **list;
size_t list_len;
code_line_info *item;
off_t start;
off_t pos;
off_t len;
char buffer[64];
uint64_t base = 0;
uint64_t offset = 0;
size_t i;
size_t k;
proc = create_x86_processor();
pos = 0;
len = 0x28;
bin_data = map_binary_file("/tmp/hello", &length);
printf(" ~~ bin_data ~~ :: %p (%d)\n", bin_data, length);
if (bin_data == NULL) return;
format = load_elf(bin_data, length);
dformat = load_dwarf(bin_data, length, format);
//comments_count = get_dwarf_comments(dformat, &comments, &offsets);
comments = NULL;
offsets = NULL;
comments_count = 0;
get_elf_symbol_comments(format, &comments, &offsets, &comments_count);
comments_list = (code_line_info **)calloc(comments_count, sizeof(code_line_info *));
for (i = 0; i < comments_count; i++)
comments_list[i] = create_code_line_info(offsets[i], NULL, strdup(comments[i]));
qsort(comments_list, comments_count, sizeof(code_line_info *), compare_code_line_info);
parts = get_elf_default_code_parts(format, &parts_count);
list = NULL;
list_len = 0;
gtk_snippet_set_format(snippet, format);
gtk_snippet_set_processor(snippet, proc);
for (i = 0; i < parts_count; i++)
{
get_bin_part_values(parts[i], &pos, &len, &base);
/*find_line_info(bin_data, &len);*/
/*
printf("Exiting...\n");
exit(0);
*/
offset = base;
for (k = 0; k < comments_count; k++)
if (comments_list[k]->offset >= base) break;
item = create_code_line_info(offset, NULL, "Simple HelloWorld !");
list = (code_line_info **)realloc(list, ++list_len * sizeof(code_line_info *));
list[list_len - 1] = item;
start = pos;
pos = 0;
while (pos < len)
{
offset = base + pos;
/* Si on a un commentaire pour cette ligne... */
if (k < comments_count && comments_list[k]->offset == offset)
{
list = (code_line_info **)realloc(list, ++list_len * sizeof(code_line_info *));
list[list_len - 1] = comments_list[k++];
}
instr = decode_instruction(proc, &bin_data[start], &pos, len, offset);
item = create_code_line_info(offset, instr, NULL);
list = (code_line_info **)realloc(list, ++list_len * sizeof(code_line_info *));
list[list_len - 1] = item;
//gtk_snippet_add_line(snippet, offset, instr, NULL);
}
/****
ret = munmap(bin_data, length);
****/
/*
gtk_snippet_build_content(snippet);
*/
}
for (i = 0; i < list_len; i++)
{
gtk_snippet_add_line(snippet, list[i]);
/* TODO: free() */
}
handle_new_exe_on_symbols_panel(panel, format);
}