summaryrefslogtreecommitdiff
path: root/src/analysis/decomp
diff options
context:
space:
mode:
Diffstat (limited to 'src/analysis/decomp')
-rw-r--r--src/analysis/decomp/Makefile.am20
-rw-r--r--src/analysis/decomp/cmerge.c135
-rw-r--r--src/analysis/decomp/cmerge.h37
-rw-r--r--src/analysis/decomp/decompiler.c247
-rw-r--r--src/analysis/decomp/decompiler.h37
-rw-r--r--src/analysis/decomp/il.c772
-rw-r--r--src/analysis/decomp/il.h50
-rw-r--r--src/analysis/decomp/reduce.c422
-rw-r--r--src/analysis/decomp/reduce.h37
9 files changed, 0 insertions, 1757 deletions
diff --git a/src/analysis/decomp/Makefile.am b/src/analysis/decomp/Makefile.am
deleted file mode 100644
index 8a9e976..0000000
--- a/src/analysis/decomp/Makefile.am
+++ /dev/null
@@ -1,20 +0,0 @@
-
-noinst_LTLIBRARIES = libanalysisdecomp.la
-
-libanalysisdecomp_la_SOURCES = \
- cmerge.h cmerge.c \
- decompiler.h decompiler.c \
- il.h il.c \
- reduce.h reduce.c
-
-libanalysisdecomp_la_LDFLAGS =
-
-
-devdir = $(includedir)/chrysalide/$(subdir:src/%=%)
-
-dev_HEADERS = $(libanalysisdecomp_la_SOURCES:%c=)
-
-
-AM_CPPFLAGS = $(LIBGTK_CFLAGS) $(LIBXML_CFLAGS)
-
-AM_CFLAGS = $(DEBUG_CFLAGS) $(WARNING_FLAGS) $(COMPLIANCE_FLAGS)
diff --git a/src/analysis/decomp/cmerge.c b/src/analysis/decomp/cmerge.c
deleted file mode 100644
index 6777263..0000000
--- a/src/analysis/decomp/cmerge.c
+++ /dev/null
@@ -1,135 +0,0 @@
-
-/* Chrysalide - Outil d'analyse de fichiers binaires
- * cmerge.c - fusion des conditions successives
- *
- * Copyright (C) 2013-2017 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 Chrysalide. If not, see <http://www.gnu.org/licenses/>.
- */
-
-
-#include "cmerge.h"
-
-
-#include "../../decomp/expr/block.h"
-#include "../../decomp/expr/comp.h"
-#include "../../decomp/expr/cond.h"
-#include "../../decomp/instr/ite.h"
-
-
-
-/* Recherche des conditions successives pouvant être fusionnées. */
-static bool track_branch_conditions(GDecInstruction *, GDecInstruction *, DecInstrVisitFlags, void *);
-
-
-
-/******************************************************************************
-* *
-* Paramètres : instr = instruction visitée. *
-* parent = instruction parente. *
-* flags = moments des appels réalisés en retour. *
-* data = adresse non utilisée ici. *
-* *
-* Description : Recherche des conditions successives pouvant être fusionnées.*
-* *
-* Retour : true afin d'aller jusqu'au terme du parcours. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool track_branch_conditions(GDecInstruction *instr, GDecInstruction *parent, DecInstrVisitFlags flags, void *data)
-{
- GDecInstruction *true_branch; /* Branche 'vrai' */
- GDecInstruction *false_branch; /* Branche 'faux' */
- GDecInstruction *first; /* Première sous-instruction */
- GCondExpression *sub_cond; /* Expression conditionnelle */
- GCondExpression *top_cond; /* Expression conditionnelle */
-
- if (!G_IS_ITE_INSTRUCTION(instr))
- goto tbc_done;
-
- g_ite_instruction_get_branches(G_ITE_INSTRUCTION(instr), &true_branch, &false_branch);
- if (false_branch != NULL) goto tbc_done;
-
- /* count(block) == 1 */
-
- first = g_expr_block_get_item(G_EXPR_BLOCK(true_branch), 0);
- if (!G_IS_ITE_INSTRUCTION(first)) goto tbc_done;
-
- printf("got one!\n");
-
- /* Récupération des informations inférieures */
-
- sub_cond = g_ite_instruction_get_condition(G_ITE_INSTRUCTION(first));
- g_object_ref(G_OBJECT(sub_cond));
-
- g_ite_instruction_get_branches(G_ITE_INSTRUCTION(first), &true_branch, &false_branch);
- if (true_branch != NULL) g_object_ref(G_OBJECT(true_branch));
- if (false_branch != NULL) g_object_ref(G_OBJECT(false_branch));
-
- /* Reconstitution d'une nouvelle instruction */
-
- top_cond = g_ite_instruction_get_condition(G_ITE_INSTRUCTION(instr));
-
- //g_cond_expression_add_condition(top_cond, sub_cond, COT_AND);
-
- //g_ite_instruction_set_branches(G_ITE_INSTRUCTION(instr), true_branch, false_branch);
-
-
-
-
- printf("got one!\n");
-
- /*
-GDecExpression *g_ite_instruction_get_condition(const GITEInstruction *instr)
-
-
-
-GDecExpression *g_cond_expression_get_expression(const GCondExpression *cond)
-
-
-void g_cond_expression_set_expression(GCondExpression *cond, GDecExpression *exp)
- */
-
-
- tbc_done:
-
- return true;
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : instr = instructions à traiter. *
-* *
-* Description : Fusionne les conditions qui s'enchaînent. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-void merge_lonely_conditions(GDecInstruction *instr)
-{
- return;
-
- g_dec_instruction_visit(instr, (dec_instr_visitor_cb)track_branch_conditions,
- DVF_EXIT, NULL);
-
-}
diff --git a/src/analysis/decomp/cmerge.h b/src/analysis/decomp/cmerge.h
deleted file mode 100644
index bd89604..0000000
--- a/src/analysis/decomp/cmerge.h
+++ /dev/null
@@ -1,37 +0,0 @@
-
-/* Chrysalide - Outil d'analyse de fichiers binaires
- * cmerge.h - prototypes pour la fusion des conditions successives
- *
- * Copyright (C) 2013-2017 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 Chrysalide. If not, see <http://www.gnu.org/licenses/>.
- */
-
-
-#ifndef _ANALYSIS_DECOMP_CMERGE_H
-#define _ANALYSIS_DECOMP_CMERGE_H
-
-
-#include "../../decomp/instruction.h"
-
-
-
-/* Fusionne les conditions qui s'enchaînent. */
-void merge_lonely_conditions(GDecInstruction *);
-
-
-
-#endif /* _ANALYSIS_DECOMP_CMERGE_H */
diff --git a/src/analysis/decomp/decompiler.c b/src/analysis/decomp/decompiler.c
deleted file mode 100644
index 03d2b70..0000000
--- a/src/analysis/decomp/decompiler.c
+++ /dev/null
@@ -1,247 +0,0 @@
-
-/* Chrysalide - Outil d'analyse de fichiers binaires
- * decompiler.c - encadrement des phases de décompilation
- *
- * Copyright (C) 2010-2017 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 Chrysalide. If not, see <http://www.gnu.org/licenses/>.
- */
-
-
-#include "decompiler.h"
-
-
-#include <malloc.h>
-#include <stdio.h>
-#include <string.h>
-
-
-#include <i18n.h>
-
-
-#include "cmerge.h"
-#include "il.h"
-#include "reduce.h"
-#include "../../decomp/output.h"
-#include "../../decomp/expr/block.h"
-#include "../../decomp/lang/java.h" /* FIXME : remme ! */
-#include "../../format/format.h"
-
-
-
-/* Construit la description d'introduction de la décompilation. */
-static void build_decomp_prologue(GCodeBuffer *, const char *);
-
-/* S'assure de la transcription de routines en expressions. */
-static void prepare_all_routines_for_decomp(const GLoadedBinary *, const char *);
-
-
-
-/******************************************************************************
-* *
-* Paramètres : buffer = tampon de destination pour le texte. *
-* filename = nom du fichier ciblé à décompiler. *
-* *
-* Description : Construit la description d'introduction de la décompilation. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static void build_decomp_prologue(GCodeBuffer *buffer, const char *filename)
-{
- GLangOutput *output; /* Modèle de sortie adéquat */
- GBufferLine *line; /* Ligne de destination */
- size_t len; /* Taille du texte */
- char *content; /* Contenu textuel d'une ligne */
-
- output = g_java_output_new();
-
- line = g_lang_output_start_comments(output, buffer);
- if (line != NULL) g_buffer_line_start_merge_at(line, BLC_PHYSICAL);
-
- /* Introduction */
-
- line = g_lang_output_continue_comments(output, buffer,
- SL(_("Binary data decompiled by Chrysalide")));
- g_buffer_line_start_merge_at(line, BLC_PHYSICAL);
-
- line = g_lang_output_continue_comments(output, buffer,
- SL(_("Chrysalide is free software - © 2008-2012 Cyrille Bagard")));
- g_buffer_line_start_merge_at(line, BLC_PHYSICAL);
-
- line = g_lang_output_continue_comments(output, buffer, NULL, 0);
- g_buffer_line_start_merge_at(line, BLC_PHYSICAL);
-
- /* Fichier */
-
- if (filename == NULL)
- filename = _("whole binary");
-
- len = strlen(_("File: ")) + strlen(filename) + 1;
- content = (char *)calloc(len, sizeof(char));
-
- snprintf(content, len, "%s%s", _("File: "), filename);
-
- line = g_lang_output_continue_comments(output, buffer, content, len - 1);
- g_buffer_line_start_merge_at(line, BLC_PHYSICAL);
-
- free(content);
-
- /* Ligne de séparation */
-
- line = g_lang_output_continue_comments(output, buffer, NULL, 0);
- g_buffer_line_start_merge_at(line, BLC_PHYSICAL);
-
- /* Conclusion */
-
- line = g_lang_output_end_comments(output, buffer);
- if (line != NULL) g_buffer_line_start_merge_at(line, BLC_PHYSICAL);
-
- g_object_unref(G_OBJECT(output));
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : binary = représentation de binaire chargé. *
-* filename = nom du fichier source à cibler. *
-* *
-* Description : S'assure de la transcription de routines en expressions. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static void prepare_all_routines_for_decomp(const GLoadedBinary *binary, const char *filename)
-{
- GExeFormat *format; /* Format du binaire fourni */
- GArchProcessor *proc; /* Architecture du binaire */
-
- GBinRoutine **routines;
- size_t count;
-
- size_t i;
-
- //GDecContext *context; /* Contexte pour la décompil. */
- GDecInstruction *dinstrs;
-
- //GArchInstruction *instrs; /* Instructions natives */
-
- //vmpa_t max; /* Première adresse à écarter */
-
- format = g_loaded_binary_get_format(binary);
- proc = NULL;//get_arch_processor_from_format(G_EXE_FORMAT(format));
-
-
- routines = NULL; //g_binary_format_get_routines(G_BIN_FORMAT(format), &count);
- count = 0;
-
-
-
- for (i = 0; i < count; i++)
- {
- /*
- context = g_arch_processor_get_decomp_context(proc);
-
- g_object_set_data(G_OBJECT(context), "format", format);
- g_object_set_data(G_OBJECT(context), "routine", routines[i]);
- g_dec_context_set_max_address(context, max);
- */
- /*
- instrs = g_binary_routine_get_instructions(routines[i]);
-
- max = g_XXX_binary_routine_get_address(routines[i])
- + g_XXX_binary_routine_get_size(routines[i]);
- */
- //printf("\n##### DECOMPILE '%s' #####\n", g_binary_routine_to_string(routines[i]));
-
-
- dinstrs = decompiled_routine_instructions(routines[i], format, proc);
-
- /*
- dinstrs = build_decompiled_block(instrs,
- g_XXX_binary_routine_get_address(routines[i]),
- max, VMPA_MAX, context);
- */
-
- //instr = g_binary_format_decompile_routine(G_BIN_FORMAT(format), routines[i], context);
-
-
- //merge_lonely_conditions(dinstrs);
-
- /////reduce_used_variables(dinstrs);
-
-
- g_expr_block_set_border_behavior(G_EXPR_BLOCK(dinstrs), BBB_FORCE_ON);
-
- g_binary_routine_set_decomp_instructions(routines[i], dinstrs);
-
-
- /*
- if (context != NULL)
- g_object_unref(context);
- */
-
- }
-
- g_object_unref(G_OBJECT(format));
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : binary = représentation de binaire chargé. *
-* filename = nom du fichier source à cibler. *
-* *
-* Description : Procède à la décompilation des routines d'un fichier donné. *
-* *
-* Retour : Tampon de code mis en place. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-GCodeBuffer *decompile_all_from_file(const GLoadedBinary *binary, const char *filename)
-{
- GCodeBuffer *result; /* Tampon constitué à renvoyer */
- GExeFormat *format; /* Format du binaire fourni */
-
- result = g_code_buffer_new(BLC_ASSEMBLY_HEAD);
-
-
- build_decomp_prologue(result, filename);
-
- //prepare_all_routines_for_decomp(binary, filename);
-
-
-
-
- format = g_loaded_binary_get_format(binary);
-
- g_binary_format_decompile(G_BIN_FORMAT(format), result, filename);
-
- g_object_unref(G_OBJECT(format));
-
- return result;
-
-}
diff --git a/src/analysis/decomp/decompiler.h b/src/analysis/decomp/decompiler.h
deleted file mode 100644
index 12a3fb4..0000000
--- a/src/analysis/decomp/decompiler.h
+++ /dev/null
@@ -1,37 +0,0 @@
-
-/* Chrysalide - Outil d'analyse de fichiers binaires
- * decompiler.h - prototypes pour l'encadrement des phases de décompilation
- *
- * Copyright (C) 2010-2017 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 Chrysalide. If not, see <http://www.gnu.org/licenses/>.
- */
-
-
-#ifndef _ANALYSIS_DECOMP_DECOMPILER_H
-#define _ANALYSIS_DECOMP_DECOMPILER_H
-
-
-#include "../binary.h"
-
-
-
-/* Procède à la décompilation des routines d'un fichier donné. */
-GCodeBuffer *decompile_all_from_file(const GLoadedBinary *, const char *);
-
-
-
-#endif /* _ANALYSIS_DECOMP_DECOMPILER_H */
diff --git a/src/analysis/decomp/il.c b/src/analysis/decomp/il.c
deleted file mode 100644
index cbc0c37..0000000
--- a/src/analysis/decomp/il.c
+++ /dev/null
@@ -1,772 +0,0 @@
-
-/* Chrysalide - Outil d'analyse de fichiers binaires
- * il.h - mise en place d'un langage intermédiaire
- *
- * Copyright (C) 2012-2017 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 Chrysalide. If not, see <http://www.gnu.org/licenses/>.
- */
-
-
-#include "il.h"
-
-
-#include <malloc.h>
-
-
-#include "../blocks/flow.h"
-#include "../blocks/virtual.h"
-#include "../../decomp/expr/block.h"
-#include "../../decomp/expr/immediate.h"
-#include "../../decomp/instr/ite.h"
-#include "../../decomp/instr/keyword.h"
-#include "../../decomp/instr/switch.h"
-
-
-
-/* --------------------- GESTION DES CONTEXTES DE DECOMPILATION --------------------- */
-
-
-/* Détermine les registres utilisés avant leur initialisation. */
-static bool track_used_registers(GFlowBlock *, BlockFollowPosition, GRAccessList **);
-
-/* Etablit le relévé des allocations de registre. */
-static void setup_awaited_regs_allocation(const GInstrBlock *, vmpa_t);
-
-/* Etablit la liste de tous les allocations attendues. */
-static bool merge_all_awaited_regs(GInstrBlock *, BlockVisitOrder, GRAccessList *);
-
-/* Met en place un contexte adapté aux sous-blocs d'un bloc. */
-static GDecContext *create_new_context_for_sub_block(GDecContext *, GInstrBlock *, GHashTable *);
-
-
-
-/* -------------------------- ENCADREMENT DES INSTRUCTIONS -------------------------- */
-
-
-/* Retrouve et rassemble les instructions décompilées. */
-static GDecInstruction *merge_decompiled_instructions(GDecInstruction *, GDecInstruction *);
-
-/* Procède à la décompilation d'un bloc déterminé. */
-static GDecInstruction *decompiled_instructions_block(GFlowBlock *, GDecContext *);
-
-/* Procède à la décompilation d'un ensemble de blocs déterminé. */
-static GDecInstruction *decompiled_instructions_blocks(GVirtualBlock *, GDecContext *);
-
-/* Procède à la décompilation d'un bloc basique quelconque. */
-static GDecInstruction *decompiled_basic_block(GInstrBlock *, GDecContext *);
-
-
-
-/* --------------------------- DECOMPILATIONS SPECIFIQUES --------------------------- */
-
-
-/* Procède à la décompilation des éléments d'un 'if then else'. */
-static void build_ite_branches(GITEInstruction *, GFlowBlock *, GDecContext *);
-
-/* Termine le traitement d'un cas de 'switch'. */
-static void close_case_decomp_instructions(GDecInstruction *, GInstrBlock *, GArchInstruction **, size_t, size_t);
-
-/* Procède à la décompilation des éléments d'un 'switch'. */
-static void build_switch_branches(GSwitchInstruction *, GFlowBlock *, GDecContext *);
-
-
-
-/* ---------------------------------------------------------------------------------- */
-/* GESTION DES CONTEXTES DE DECOMPILATION */
-/* ---------------------------------------------------------------------------------- */
-
-
-/******************************************************************************
-* *
-* Paramètres : block = bloc d'instructions visité. *
-* pos = indication sur la position du parcours. *
-* needed = suivi de l'usage des registres entre les blocs. *
-* *
-* Description : Détermine les registres utilisés avant leur initialisation. *
-* *
-* Retour : true pour mener un parcours complet. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool track_used_registers(GFlowBlock *block, BlockFollowPosition pos, GRAccessList **needed)
-{
- GRAccessList *old; /* Ancienne liste remplacée */
- const GRAccessList *accesses; /* Accès divers aux registres */
- size_t count; /* Taille d'un parcours */
- GRAccessList *awaited; /* Satisfactions des besoins */
- reg_access *access; /* Accès à un registre */
- size_t i; /* Boucle de parcours */
- reg_access *found; /* Besoin trouvé ou non */
-
- switch (pos)
- {
- case BFP_ENTER:
- g_object_set_data(G_OBJECT(block), "needed_regs", *needed);
- break;
-
- case BFP_FOLLOW:
- *needed = g_raccess_list_new();
- break;
-
- case BFP_BACK:
-
- old = *needed;
- *needed = G_RACCESS_LIST(g_object_get_data(G_OBJECT(block), "needed_regs"));
-
- g_raccess_list_merge(*needed, old);
- g_object_unref(G_OBJECT(old));
-
- break;
-
- case BFP_EXIT:
-
- g_object_set_data(G_OBJECT(block), "needed_regs", NULL);
-
- accesses = g_flow_block_list_regs_accesses(block);
- count = g_raccess_list_count(accesses);
-
- awaited = g_flow_block_list_awaited_regs(block);
-
- for (i = 0; i < count; i++)
- {
- access = g_raccess_list_get(accesses, i);
-
- /* Enregistrement des contributions possibles */
-
- found = g_raccess_list_find(*needed, access->reg);
-
- if (found != NULL)
- {
- /**
- * Si un autre bloc avait déjà un besoin, on ne prend note
- * qu'une seule fois !
- */
- if (g_raccess_list_find(awaited, access->reg) == NULL)
- g_raccess_list_add(awaited, access);
-
- g_raccess_list_delete(*needed, found);
-
- }
-
- /* Ajoute les besoins du bloc */
- if (access->first_access == RAT_READ)
- g_raccess_list_add(*needed, access);
-
- }
-
- /*
- do
- {
- vmpa_t start, end;
-
- g_flow_block_get_boundary_addresses(block, &start, &end);
-
- printf(" -> flow (%d) : 0x%08x - 0x%08x | needed = %zu - provided = %zu\n",
- pos, start, end,
- g_raccess_list_count(*needed), g_raccess_list_count(awaited));
-
- }
- while (0);
- */
-
- break;
-
- }
-
- return true;
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : list = ensemble des instructions d'assemblage à traiter. *
-* start = adresse de départ de la routine visée. *
-* *
-* Description : Etablit le relévé des allocations de registre. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static void setup_awaited_regs_allocation(const GInstrBlock *list, vmpa_t start)
-{
- GInstrBlock *first; /* Bloc de départ du flot */
- GRAccessList *needed; /* Registres inter-blocs */
-
- first = g_instr_block_find_by_addr(list, start, true);
-
- needed = g_raccess_list_new();
-
- g_flow_block_follow(G_FLOW_BLOCK(first), list, BFP_ENTER | BFP_EXIT | BFP_EXIT,
- (flow_block_follow_cb)track_used_registers, &needed);
-
- g_object_unref(G_OBJECT(needed));
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : block = bloc d'instructions concerné par la visite. *
-* order = position dans la visite. *
-* list = liste à compléter. *
-* *
-* Description : Etablit la liste de tous les allocations attendues. *
-* *
-* Retour : true pour parcourir tous les blocs. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool merge_all_awaited_regs(GInstrBlock *block, BlockVisitOrder order, GRAccessList *list)
-{
- const GRAccessList *awaited; /* Registres conséquents */
-
- if (G_IS_FLOW_BLOCK(block))
- {
- awaited = g_flow_block_list_regs_accesses(G_FLOW_BLOCK(block));
- g_raccess_list_merge(list, awaited);
- }
-
- return true;
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : ctx = contexte de décompilation courant. *
-* block = block regroupant les branches de division. *
-* shared = liste des allocations passées de registres attendus.*
-* *
-* Description : Met en place un contexte adapté aux sous-blocs d'un bloc. *
-* *
-* Retour : Nouveau contexte près à emploi. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static GDecContext *create_new_context_for_sub_block(GDecContext *ctx, GInstrBlock *block, GHashTable *shared)
-{
- GDecContext *result; /* Contexte à retourner */
- GRAccessList *list; /* Allocations attendues */
-
- result = g_dec_context_dup(ctx);
-
- list = g_raccess_list_new();
- g_instr_block_visit(block, (instr_block_visitor_cb)merge_all_awaited_regs, list);
- g_dec_context_set_awaited(result, list);
- g_object_unref(G_OBJECT(list));
-
- g_dec_context_set_shared_allocs(result, shared);
-
- return result;
-
-}
-
-
-
-/* ---------------------------------------------------------------------------------- */
-/* ENCADREMENT DES INSTRUCTIONS */
-/* ---------------------------------------------------------------------------------- */
-
-
-/******************************************************************************
-* *
-* Paramètres : group = groupe d'instructions à compléter ou constituer. *
-* list = liste d'instructions à intégrer. *
-* *
-* Description : Retrouve et rassemble les instructions décompilées. *
-* *
-* Retour : Groupe fourni ou nouveau groupe créé pour l'occasion. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static GDecInstruction *merge_decompiled_instructions(GDecInstruction *group, GDecInstruction *list)
-{
- GExprBlock *block; /* Autre vision du groupe */
- GDecInstruction *iter; /* Boucle de parcours */
-
- if (group == NULL) block = NULL;
- else block = G_EXPR_BLOCK(group);
-
- for (iter = list;
- iter != NULL;
- iter = g_dec_instruction_get_next_iter(list, iter))
- {
- if (block == NULL)
- block = G_EXPR_BLOCK(g_expr_block_new(iter));
- else
- g_expr_block_add_item(block, iter);
-
- }
-
- return G_DEC_INSTRUCTION(block);
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : block = ensemble des instructions d'assemblage à traiter. *
-* ctx = contexte de soutien à associer à l'opération. *
-* *
-* Description : Procède à la décompilation d'un bloc déterminé. *
-* *
-* Retour : Instructions créées et enregistrées, ou NULL si erreur. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static GDecInstruction *decompiled_instructions_block(GFlowBlock *block, GDecContext *ctx)
-{
- GArchInstruction *instrs; /* Liste d'instructions natives*/
- GArchInstruction *first; /* Première instruction du lot */
- vmpa_t max; /* Adresse de fin du bloc */
- GArchInstruction *iter; /* Boucle de parcours #1 */
- GDecInstruction *decomp; /* Dernier résultat de décomp. */
-
- instrs = NULL; // FIXME g_flow_block_get_all_instructions_list(block);
- g_flow_block_get_boundary(block, &first, NULL);
-
- g_flow_block_get_boundary_addresses(block, NULL, &max);
- max++;
-
- /* Décompilation du corps du bloc */
-
- for (iter = first;
- iter != NULL;
- iter = g_arch_instruction_get_next_iter(instrs, iter, max))
- {
- decomp = g_arch_instruction_decompile(iter, ctx);
- }
-
- /* Post-traitement selon les types de lien */
-
- /* if ... then ... else ... */
- if (G_IS_ITE_INSTRUCTION(decomp))
- build_ite_branches(G_ITE_INSTRUCTION(decomp), block, ctx);
-
- /* switch ... case ... */
- else if (G_IS_SWITCH_INSTRUCTION(decomp))
- build_switch_branches(G_SWITCH_INSTRUCTION(decomp), block, ctx);
-
- /* Renvoi des instructions mises en place */
-
- return g_dec_context_get_decomp_instrs(ctx);
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : block = ensemble des instructions d'assemblage à traiter. *
-* ctx = contexte de soutien à associer à l'opération. *
-* *
-* Description : Procède à la décompilation d'un ensemble de blocs déterminé. *
-* *
-* Retour : Instructions créées et enregistrées, ou NULL si erreur. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static GDecInstruction *decompiled_instructions_blocks(GVirtualBlock *block, GDecContext *ctx)
-{
- GDecInstruction *result; /* Instructions à renvoyer */
- size_t count; /* Nombre de sous-blocs */
- size_t i; /* Boucle de parcours */
- GInstrBlock *sub_block; /* Sous-bloc à traiter */
- GDecInstruction *dinstrs; /* Instructions décompilées */
-
- result = NULL;
-
- count = g_virtual_block_count_children(block);
-
- for (i = 0; i < count; i++)
- {
- sub_block = g_virtual_block_get_child(block, i);
-
- /**
- * Les groupes de blocs sont forcément rattachés à une instruction,
- * donc ils sont décompilés depuis ces instructions, pas ici.
- */
- if (!G_IS_FLOW_BLOCK(sub_block)) continue;
-
- /* FIXME */
- //g_dec_context_set_decomp_instrs(ctx, NULL);
-
- dinstrs = decompiled_instructions_block(G_FLOW_BLOCK(sub_block), ctx);
- //result = merge_decompiled_instructions(result, dinstrs);
-
- }
-
- result = merge_decompiled_instructions(result, dinstrs);
-
- return result;
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : block = ensemble des instructions d'assemblage à traiter. *
-* ctx = contexte de soutien à associer à l'opération. *
-* *
-* Description : Procède à la décompilation d'un bloc basique quelconque. *
-* *
-* Retour : Instructions créées et enregistrées, ou NULL si erreur. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static GDecInstruction *decompiled_basic_block(GInstrBlock *block, GDecContext *ctx)
-{
- GDecInstruction *result; /* Instructions à retourner */
-
- if (G_IS_VIRTUAL_BLOCK(block))
- result = decompiled_instructions_blocks(G_VIRTUAL_BLOCK(block), ctx);
-
- else
- {
- result = decompiled_instructions_block(G_FLOW_BLOCK(block), ctx);
-
- if (!G_IS_EXPR_BLOCK(result))
- result = merge_decompiled_instructions(NULL, result);
-
- }
-
- return result;
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : routine = routine dont le corps est à traiter. *
-* format = format du binaire contenant les instructions. *
-* proc = architecture du code machine. *
-* *
-* Description : Procède à la décompilation d'une routinée déterminée. *
-* *
-* Retour : Instructions créées ou NULL si erreur. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-GDecInstruction *decompiled_routine_instructions(GBinRoutine *routine, GExeFormat *format, GArchProcessor *proc)
-{
- GDecInstruction *result; /* Instructions à retourner */
- GDecContext *context; /* Contexte pour la décompil. */
- GInstrBlock *blocks; /* Blocs basiques de routine */
- const mrange_t *range; /* Emplacement du symbole */
-
- context = g_arch_processor_get_decomp_context(proc);
- g_dec_context_set_info(context, routine, format);
-
- blocks = g_binary_routine_get_basic_blocks(routine);
-
- range = g_binary_symbol_get_range(G_BIN_SYMBOL(routine));
- setup_awaited_regs_allocation(blocks, get_mrange_addr(range)->physical);
-
- result = decompiled_basic_block(blocks, context);
-
- g_object_unref(context);
-
- return result;
-
-}
-
-
-
-/* ---------------------------------------------------------------------------------- */
-/* DECOMPILATIONS SPECIFIQUES */
-/* ---------------------------------------------------------------------------------- */
-
-
-/******************************************************************************
-* *
-* Paramètres : decomp = instruction 'if ... then ... else' à compléter. *
-* block = ensemble des instructions d'assemblage traitées. *
-* ctx = contexte de soutien à associer à l'opération. *
-* *
-* Description : Procède à la décompilation des éléments d'un 'if then else'. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static void build_ite_branches(GITEInstruction *decomp, GFlowBlock *block, GDecContext *ctx)
-{
- GArchInstruction *last; /* Dernière instruction du lot */
- GInstrBlock *sub_parent; /* Groupe des sous-branches */
- GHashTable *sub_shared; /* Allocations communes */
- GDecContext *sub_ctx; /* Sous-contexte pour branche */
- GDecInstruction *true_dinstr; /* Décompilation 'cond vraie' */
- GDecInstruction *false_dinstr; /* Décompilation 'cond fausse' */
- GArchInstruction *next; /* Instruction de branchement */
- vmpa_t next_addr; /* Adresse de cette instruct° */
- GInstrBlock *next_block; /* Sous-bloc basique direct */
-
- g_flow_block_get_boundary(block, NULL, &last);
-
- sub_parent = g_instr_block_get_links_block(G_INSTR_BLOCK(block));
- sub_shared = g_hash_table_new_full((GHashFunc)g_arch_register_hash,
- (GEqualFunc)NULL,//g_arch_register_equal,
- g_object_unref, g_object_unref);
-
- true_dinstr = NULL;
- false_dinstr = NULL;
-
- /* Branche 'true' */
- next = g_arch_instruction_get_given_destination(last, ILT_JUMP_IF_TRUE);
- if (next != NULL)
- {
- g_arch_instruction_get_location(next, NULL, NULL, &next_addr);
- next_block = g_instr_block_find_by_addr(sub_parent, next_addr, false);
-
- if (next_block != NULL)
- {
- sub_ctx = create_new_context_for_sub_block(ctx, next_block, sub_shared);
- true_dinstr = decompiled_basic_block(next_block, sub_ctx);
- g_dec_context_spread_allocated_shared_regs(ctx, sub_ctx);
- g_object_unref(G_OBJECT(sub_ctx));
- }
-
- g_object_unref(G_OBJECT(next));
-
- }
-
- /* Branche 'false' */
- next = g_arch_instruction_get_given_destination(last, ILT_JUMP_IF_FALSE);
- if (next != NULL)
- {
- g_arch_instruction_get_location(next, NULL, NULL, &next_addr);
- next_block = g_instr_block_find_by_addr(sub_parent, next_addr, false);
-
- if (next_block != NULL)
- {
- sub_ctx = create_new_context_for_sub_block(ctx, next_block, sub_shared);
- false_dinstr = decompiled_basic_block(next_block, sub_ctx);
- g_dec_context_spread_allocated_shared_regs(ctx, sub_ctx);
- g_object_unref(G_OBJECT(sub_ctx));
- }
-
- g_object_unref(G_OBJECT(next));
-
- }
-
- g_ite_instruction_set_branches(decomp, true_dinstr, false_dinstr);
-
- g_hash_table_unref(sub_shared);
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : case_dinstr = instructions d'un cas de 'switch' décompilées. *
-* case_block = bloc d'instructions assembleur correspondantes.*
-* cases = listes des instructions des différents cas. *
-* current = indice du cas courant. *
-* ccount = nombre de cas présents dans le 'switch'. *
-* *
-* Description : Termine le traitement d'un cas de 'switch'. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static void close_case_decomp_instructions(GDecInstruction *case_dinstr, GInstrBlock *case_block, GArchInstruction **cases, size_t current, size_t ccount)
-{
- vmpa_t *cases_addr; /* Adresses des cas du 'switch'*/
- size_t i; /* Boucle de parcours #1 */
- GInstrBlock **leafs; /* Blocs terminaux du cas */
- size_t lcount; /* Nombre de blocs terminaux */
- vmpa_t common_addr; /* Adresse commune de suite */
- bool is_common; /* Suite commune ? */
- GArchInstruction *last; /* Dernière instruction de bloc*/
- size_t dcount; /* Nombre de liens de dest. */
- size_t j; /* Boucle de parcours #2 */
- const instr_link_t *dest; /* Instr. visée par une autre */
- vmpa_t addr; /* Adresse d'une instruction */
- bool jump_to_case; /* Suite dans le cas suivant ? */
-
- /* Etablit la liste des adresses des cas */
-
- cases_addr = (vmpa_t *)calloc(ccount, sizeof(vmpa_t));
-
- for (i = 0; i < ccount; i++)
- g_arch_instruction_get_location(cases[i], NULL, NULL, &cases_addr[i]);
-
- /* Récupère la (les) fin(s) du cas présent */
-
- leafs = NULL;
- lcount = 0;
-
- g_instr_block_list_leafs_blocks(case_block, &leafs, &lcount);
-
- /* Procède à une première analyse de la suite */
-
- common_addr = VMPA_MAX;
- is_common = true;
-
- for (i = 0; i < lcount && is_common; i++)
- {
- g_flow_block_get_boundary(G_FLOW_BLOCK(leafs[i]), NULL, &last);
-
- g_arch_instruction_lock_dest(last);
- dcount = g_arch_instruction_count_destinations(last);
-
- for (j = 0; j < dcount && is_common; j++)
- {
- dest = g_arch_instruction_get_destination(last, j);
-
- g_arch_instruction_get_location(dest->linked, NULL, NULL, &addr);
-
- if (common_addr == VMPA_MAX)
- common_addr = addr;
- else
- is_common = (common_addr == addr);
-
- unref_instr_link(dest);
-
- }
-
- g_arch_instruction_unlock_dest(last);
-
- }
-
- /* La sortie du cas est unique ! */
- if (is_common)
- {
- if (common_addr == VMPA_MAX)
- goto ecdi_exit;
-
- jump_to_case = false;
-
- for (i = 0; i < ccount && !jump_to_case; i++)
- if (i != current)
- jump_to_case = (cases_addr[i] == common_addr);
-
- if (!jump_to_case)
- g_expr_block_add_item(G_EXPR_BLOCK(case_dinstr), g_keyword_instruction_new(DKW_BREAK));
-
- }
-
- /* ... ou il faut suivre les différentes branches... */
- else
- {
-
- /* TODO */
-
- }
-
- ecdi_exit:
-
- free(cases_addr);
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : decomp = instruction 'switch' à compléter. *
-* block = ensemble des instructions d'assemblage traitées. *
-* ctx = contexte de soutien à associer à l'opération. *
-* *
-* Description : Procède à la décompilation des éléments d'un 'switch'. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static void build_switch_branches(GSwitchInstruction *decomp, GFlowBlock *block, GDecContext *ctx)
-{
-#if 0
-
- GArchInstruction *last; /* Dernière instruction du lot */
- GInstrBlock *sub_parent; /* Groupe des sous-branches */
- GHashTable *sub_shared; /* Allocations communes */
- GDecContext *sub_ctx; /* Sous-contexte pour branche */
- vmpa_t next_addr; /* Adresse de cette instruct° */
- GInstrBlock *next_block; /* Sous-bloc basique direct */
- GArchInstruction **dests; /* Instr. visée par une autre */
- link_extra_info *info; /* Compléments pour les liens */
- size_t dcount; /* Nombre de liens de dest. */
- size_t i; /* Boucle de parcours #2 */
- GDecInstruction *case_dinstr; /* Décompilation 'case' */
- GDecExpression *case_value; /* Valeur d'aiguillage */
-
- g_flow_block_get_boundary(block, NULL, &last);
-
- sub_parent = g_instr_block_get_links_block(G_INSTR_BLOCK(block));
- sub_shared = g_hash_table_new_full((GHashFunc)g_arch_register_hash,
- (GEqualFunc)g_arch_register_equal,
- g_object_unref, g_object_unref);
-
- dcount = g_arch_instruction_get_destinations(last, &dests, NULL, &info);
-
- for (i = 0; i < dcount; i++)
- {
- g_arch_instruction_get_location(dests[i], NULL, NULL, &next_addr);
- next_block = g_instr_block_find_by_addr(sub_parent, next_addr, false);
-
- if (next_block != NULL)
- {
- sub_ctx = create_new_context_for_sub_block(ctx, next_block, sub_shared);
- case_dinstr = decompiled_basic_block(next_block, sub_ctx);
- g_dec_context_spread_allocated_shared_regs(ctx, sub_ctx);
- g_object_unref(G_OBJECT(sub_ctx));
-
- close_case_decomp_instructions(case_dinstr, next_block, dests, i, dcount);
- g_expr_block_set_border_behavior(G_EXPR_BLOCK(case_dinstr), BBB_FORCE_OFF);
-
- if (info[i].imm != NULL)
- {
- case_value = G_DEC_EXPRESSION(g_imm_expression_new(info[i].imm));
- g_switch_instruction_add_case(G_SWITCH_INSTRUCTION(decomp),
- case_value, case_dinstr, next_addr);
- }
- else g_switch_instruction_set_default_case(G_SWITCH_INSTRUCTION(decomp),
- case_dinstr);
-
- }
-
- }
-
- g_hash_table_unref(sub_shared);
-#endif
-}
diff --git a/src/analysis/decomp/il.h b/src/analysis/decomp/il.h
deleted file mode 100644
index ed901e6..0000000
--- a/src/analysis/decomp/il.h
+++ /dev/null
@@ -1,50 +0,0 @@
-
-/* Chrysalide - Outil d'analyse de fichiers binaires
- * il.h - prototypes pour la mise en place d'un langage intermédiaire
- *
- * Copyright (C) 2012-2017 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 Chrysalide. If not, see <http://www.gnu.org/licenses/>.
- */
-
-
-#ifndef _ANALYSIS_DECOMP_IL_H
-#define _ANALYSIS_DECOMP_IL_H
-
-
-#include "../routine.h"
-#include "../../arch/processor.h"
-
-
-
-/* Procède à la décompilation d'une routinée déterminée. */
-GDecInstruction *decompiled_routine_instructions(GBinRoutine *, GExeFormat *, GArchProcessor *);
-
-
-
-
-#if 0
-#include "../../arch/instruction.h"
-#include "../../decomp/instruction.h"
-
-
-
-/* Procède à la décompilation basique d'un bloc déterminé. */
-GDecInstruction *build_decompiled_block(GArchInstruction *, vmpa_t, vmpa_t, vmpa_t, GDecContext *);
-#endif
-
-
-#endif /* _ANALYSIS_DECOMP_IL_H */
diff --git a/src/analysis/decomp/reduce.c b/src/analysis/decomp/reduce.c
deleted file mode 100644
index b630bc7..0000000
--- a/src/analysis/decomp/reduce.c
+++ /dev/null
@@ -1,422 +0,0 @@
-
-/* Chrysalide - Outil d'analyse de fichiers binaires
- * reduce.c - réduction de l'usage des [pseudo]-registres
- *
- * Copyright (C) 2012-2017 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 Chrysalide. If not, see <http://www.gnu.org/licenses/>.
- */
-
-
-#include "reduce.h"
-
-
-#include <malloc.h>
-
-
-#include "../../decomp/expr/assign.h"
-#include "../../decomp/expr/block.h"
-#include "../../decomp/expr/immediate.h"
-#include "../../decomp/expr/pseudo.h"
-
-
-
-/* ------------------------ DECOMPTE DES USAGES DE VARIABLES ------------------------ */
-
-
-/* Mémorisation de l'usage d'une variable */
-typedef struct _assigned_value
-{
- GDecInstruction *value; /* Valeur de cette variable */
-
- GDecInstruction *assign; /* Lieu d'assignation */
- GDecInstruction *parent; /* Zone à nettoyer au besoin */
-
- size_t used_counter; /* Décompte des utilisations */
- size_t shared_counter; /* Partage entre tables */
-
-} assigned_value;
-
-
-/* Conserve tous les éléments concernant une assignation. */
-static assigned_value *memorize_assigned_value(GDecInstruction *, GDecInstruction *, GDecInstruction *);
-
-/* Met (éventuellement) fin à un suivi d'assignation. */
-static void forget_assigned_value(assigned_value *);
-
-/* Constitue un lieu de conservation d'associations pour bloc. */
-static GHashTable *create_tracking_table(GHashTable *);
-
-/* Met à jour les décomptes des différentes assignations. */
-static void merge_usages(GHashTable *, GHashTable *);
-
-/* Prend notre d'une assignation d'une valeur à une variable. */
-static void note_new_variable(GHashTable *, GDecInstruction *, assigned_value *);
-
-
-
-/* ---------------------- VISITES DES INSTRUCTIONS DECOMPILEES ---------------------- */
-
-
-/* Prend note de tous les accès aux différentes variables. */
-static void collect_used_variables(GDecInstruction *, GDecInstruction *, GHashTable *);
-
-/* Procède au remplacement de variables à rejeter. */
-static void replace_useless_variable(GDecInstruction *, GHashTable *);
-
-/* Visite un groupe d'instructions décompilées. */
-static bool visit_instructions_block(GDecInstruction *, GDecInstruction *, DecInstrVisitFlags, GHashTable **);
-
-
-
-/* ---------------------------------------------------------------------------------- */
-/* DECOMPTE DES USAGES DE VARIABLES */
-/* ---------------------------------------------------------------------------------- */
-
-
-/******************************************************************************
-* *
-* Paramètres : value = nouvelle valeur assignée à suivre. *
-* assign = instruction où la valeur est assignée. *
-* parent = bloc d'instructions à soulager en cas d'effacement. *
-* table = mémorisation des usages de variables. *
-* *
-* Description : Conserve tous les éléments concernant une assignation. *
-* *
-* Retour : Structure mise en place. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static assigned_value *memorize_assigned_value(GDecInstruction *value, GDecInstruction *assign, GDecInstruction *parent)
-{
- assigned_value *result; /* Mémorisation à retourner */
-
- result = (assigned_value *)calloc(1, sizeof(assigned_value));
-
- g_object_ref(G_OBJECT(value));
- result->value = value;
-
- g_object_ref(G_OBJECT(assign));
- g_object_ref(G_OBJECT(parent));
- result->assign = assign;
- result->parent = parent;
-
- result->used_counter = 0;
- result->shared_counter = 0;
-
- return result;
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : val = structure à libérer de la mémoire. *
-* *
-* Description : Met (éventuellement) fin à un suivi d'assignation. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static void forget_assigned_value(assigned_value *val)
-{
- if (--val->shared_counter == 0)
- {
- g_object_unref(G_OBJECT(val->value));
-
- g_object_unref(G_OBJECT(val->assign));
- g_object_unref(G_OBJECT(val->parent));
-
- free(val);
-
- }
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : parent = éventuel bloc d'instructions parent ou NULL. *
-* *
-* Description : Constitue un lieu de conservation d'associations pour bloc. *
-* *
-* Retour : Table prête à emploi. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static GHashTable *create_tracking_table(GHashTable *parent)
-{
- GHashTable *result; /* Création à retourner */
- GHashTableIter iter; /* Boucle de parcours */
- GDecInstruction *var; /* Variable assignée */
- assigned_value *val; /* Valeur assignée */
-
- result = g_hash_table_new_full(NULL, NULL, g_object_unref,
- (GDestroyNotify)forget_assigned_value);
-
- if (parent != NULL)
- {
- g_hash_table_iter_init(&iter, parent);
-
- while (g_hash_table_iter_next(&iter, (gpointer *)&var, (gpointer *)&val))
- note_new_variable(result, var, val);
-
- }
-
- return result;
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : table = conservation des associations variable <-> valeur. *
-* sub = conservation issue d'un sous bloc. *
-* *
-* Description : Met à jour les décomptes des différentes assignations. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static void merge_usages(GHashTable *table, GHashTable *sub)
-{
- GHashTableIter iter; /* Boucle de parcours */
- GDecInstruction *var; /* Variable assignée */
- assigned_value *val; /* Valeur assignée */
- assigned_value *update; /* Même valeur, mise à jour */
-
- g_hash_table_iter_init(&iter, table);
-
- while (g_hash_table_iter_next(&iter, (gpointer *)&var, (gpointer *)&val))
- {
- update = (assigned_value *)g_hash_table_lookup(sub, var);
-
- val->used_counter = update->used_counter;
-
- g_hash_table_remove(sub, var);
-
- }
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : table = conservation des associations variable <-> valeur. *
-* var = variable en place à suivre. *
-* val = valeur associée. *
-* *
-* Description : Prend notre d'une assignation d'une valeur à une variable. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static void note_new_variable(GHashTable *table, GDecInstruction *var, assigned_value *val)
-{
- g_object_ref(G_OBJECT(var));
- val->shared_counter++;
-
- g_hash_table_insert(table, var, val);
-
-}
-
-
-
-/* ---------------------------------------------------------------------------------- */
-/* VISITES DES INSTRUCTIONS DECOMPILEES */
-/* ---------------------------------------------------------------------------------- */
-
-
-/******************************************************************************
-* *
-* Paramètres : instr = instruction visitée. *
-* parent = instruction parente. *
-* table = mémorisation des usages de variables. *
-* *
-* Description : Prend note de tous les accès aux différentes variables. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static void collect_used_variables(GDecInstruction *instr, GDecInstruction *parent, GHashTable *table)
-{
- GAssignExpression *assign; /* Instruction d'assignation */
- GDecInstruction *dest; /* Variable de destination */
- bool valid; /* Destination valide ? */
- GDecInstruction *src; /* Valeur de source */
- assigned_value *value; /* Valeur et ses informations */
-
- if (G_IS_ASSIGN_EXPRESSION(instr))
- {
- assign = G_ASSIGN_EXPRESSION(instr);
-
- dest = G_DEC_INSTRUCTION(g_assign_expression_get_dest(assign));
-
- if (G_IS_PSEUDO_REGISTER(dest))
- valid = (g_pseudo_register_get_usage(G_PSEUDO_REGISTER(dest)) == PRU_LOCAL);
- else
- valid = true;
-
- if (valid)
- {
- src = G_DEC_INSTRUCTION(g_assign_expression_get_src(assign));
-
- value = memorize_assigned_value(src, instr, parent);
- note_new_variable(table, dest, value);
-
- }
-
- }
- else
- {
- value = (assigned_value *)g_hash_table_lookup(table, instr);
-
- if (value != NULL)
- value->used_counter++;
-
- }
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : block = bloc d'instructions décompilées à traiter. *
-* table = mémorisation des usages de variables. *
-* *
-* Description : Procède au remplacement de variables à rejeter. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static void replace_useless_variable(GDecInstruction *block, GHashTable *table)
-{
- GHashTableIter iter; /* Boucle de parcours */
- GDecInstruction *var; /* Variable assignée */
- assigned_value *val; /* Valeur assignée */
- bool replace; /* Ordre de remplacement */
-
- g_hash_table_iter_init(&iter, table);
-
- while (g_hash_table_iter_next(&iter, (gpointer *)&var, (gpointer *)&val))
- {
- replace = (G_IS_IMM_EXPRESSION(val->value) && (val->used_counter > 1));
- replace |= (val->used_counter == 2); /* 1 assign. + 1 usage */
-
- if (!replace) continue;
-
- g_expr_block_delete_item(G_EXPR_BLOCK(val->parent), val->assign);
-
- g_dec_instruction_replace(block, var, val->value);
-
- }
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : instr = instruction visitée. *
-* parent = instruction parente. *
-* flags = moments des appels réalisés en retour. *
-* table = mémorisation des usages de variables. *
-* *
-* Description : Visite un groupe d'instructions décompilées. *
-* *
-* Retour : true afin d'aller jusqu'au terme du parcours. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool visit_instructions_block(GDecInstruction *instr, GDecInstruction *parent, DecInstrVisitFlags flags, GHashTable **table)
-{
- GHashTable *sub; /* Associations pour sous-bloc */
-
- if (G_IS_EXPR_BLOCK(instr))
- {
- /* Entrée dans un bloc : on crée un contexte propre au bloc ! */
- if (flags == DVF_ENTER)
- {
- g_object_set_data(G_OBJECT(instr), "var_collect", *table);
- *table = create_tracking_table(*table);
- }
-
- /* Sortie du bloc : actions locales + purge ! */
- else if (flags == DVF_EXIT)
- {
- sub = *table;
- *table = (GHashTable *)g_object_get_data(G_OBJECT(instr), "var_collect");
-
- if (*table != NULL)
- merge_usages(*table, sub);
-
- replace_useless_variable(instr, sub);
- g_hash_table_unref(sub);
-
- }
-
- }
-
- else if (flags == DVF_ENTER)
- collect_used_variables(instr, parent, *table);
-
- return true;
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : instr = instructions à traiter. *
-* *
-* Description : Réduit le nombre de variables utilisées, via propagations. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-void reduce_used_variables(GDecInstruction *instr)
-{
- GHashTable *table;
-
- table = NULL;
-
- g_dec_instruction_visit(instr, (dec_instr_visitor_cb)visit_instructions_block,
- DVF_ENTER | DVF_EXIT, &table);
-
-}
diff --git a/src/analysis/decomp/reduce.h b/src/analysis/decomp/reduce.h
deleted file mode 100644
index fe5fa99..0000000
--- a/src/analysis/decomp/reduce.h
+++ /dev/null
@@ -1,37 +0,0 @@
-
-/* Chrysalide - Outil d'analyse de fichiers binaires
- * reduce.h - prototypes pour la réduction de l'usage des [pseudo]-registres
- *
- * Copyright (C) 2012-2017 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 Chrysalide. If not, see <http://www.gnu.org/licenses/>.
- */
-
-
-#ifndef _ANALYSIS_DECOMP_REDUCE_H
-#define _ANALYSIS_DECOMP_REDUCE_H
-
-
-#include "../../decomp/instruction.h"
-
-
-
-/* Réduit le nombre de variables utilisées, via propagations. */
-void reduce_used_variables(GDecInstruction *instr);
-
-
-
-#endif /* _ANALYSIS_DECOMP_REDUCE_H */