/* Chrysalide - Outil d'analyse de fichiers binaires
* instruction.c - gestion générique des instructions
*
* Copyright (C) 2008-2025 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 .
*/
#include "instruction.h"
#include
#include
#include
#include
#include "instruction-int.h"
#include "../common/leb128.h"
#include "../core/logs.h"
#include "../core/processors.h"
#include "../glibext/serialize-int.h"
/* ----------------------- DEFINITION GENERIQUE D'INSTRUCTION ----------------------- */
/* Initialise la classe générique des instructions. */
static void g_arch_instruction_class_init(GArchInstructionClass *);
/* Procède à l'initialisation de l'interface de sérialisation. */
static void g_arch_instruction_serializable_object_iface_init(GSerializableObjectInterface *);
/* Initialise une instance d'opérande d'architecture. */
static void g_arch_instruction_init(GArchInstruction *);
/* Supprime toutes les références externes. */
static void g_arch_instruction_dispose(GObject *);
/* Procède à la libération totale de la mémoire. */
static void g_arch_instruction_finalize(GObject *);
/* ------------------- DEFINITION DES LIAISONS ENTRE INSTRUCTIONS ------------------- */
#define COMPACT_INS_LINK_MASK_DIR (1ul << (__WORDSIZE - 1))
#define COMPACT_INS_LINK_MASK_TYPE 0xf
#define COMPACT_INS_LINK_MASK (COMPACT_INS_LINK_MASK_DIR | COMPACT_INS_LINK_MASK_TYPE)
#define COMPACT_INS_LINK_FROM (0ul << (__WORDSIZE - 1))
#define COMPACT_INS_LINK_TO (1ul << (__WORDSIZE - 1))
#define COMPACT_INS_LINK_DIR(cl) (cl & COMPACT_INS_LINK_MASK_DIR)
#define COMPACT_INS_LINK_PTR(cl) ((GArchInstruction *)(cl & ~COMPACT_INS_LINK_MASK))
#define COMPACT_INS_LINK_TYPE(cl) (cl & COMPACT_INS_LINK_MASK_TYPE)
#define MAKE_COMPACT_INS_LINK(d, i, t) \
(compact_ins_link_t)(d | (unsigned long)i | t)
/* Détermine si un type de lien existe dans une instruction. */
static bool _g_arch_instruction_has_link(const GArchInstruction *, compact_ins_link_t, InstructionLinkType);
/* Détermine si un lien existe entre deux instructions. */
static bool _g_arch_instruction_has_link_with(const GArchInstruction *, compact_ins_link_t, const GArchInstruction *);
/* Fournit la quantité d'instructions pointant vers une autre. */
static size_t _g_arch_instruction_count_links(const GArchInstruction *, compact_ins_link_t);
/* Fournit les détails d'un lien donné avec une instruction. */
static GArchInstruction *_g_arch_instruction_get_linked_instruction(const GArchInstruction *, size_t, compact_ins_link_t, InstructionLinkType *);
/* ------------------- MECANISMES DE CONSERVATION ET RESTAURATION ------------------- */
/* Charge un objet depuis un flux de données. */
static bool g_arch_instruction_load(GSerializableObject *, GObjectStorage *, int);
/* Sauvegarde un objet dans un flux de données. */
static bool g_arch_instruction_store(const GSerializableObject *, GObjectStorage *, int);
/* ---------------------------------------------------------------------------------- */
/* DEFINITION GENERIQUE D'INSTRUCTION */
/* ---------------------------------------------------------------------------------- */
/* Indique le type défini pour une instruction d'architecture. */
G_DEFINE_TYPE_WITH_CODE(GArchInstruction, g_arch_instruction, G_TYPE_THICK_OBJECT,
G_IMPLEMENT_INTERFACE(G_TYPE_SERIALIZABLE_OBJECT, g_arch_instruction_serializable_object_iface_init)
G_IMPLEMENT_INTERFACE_IF_SYM(g_token_generator_get_type, g_arch_instruction_ui_token_generator_iface_init));
/******************************************************************************
* *
* Paramètres : klass = classe à initialiser. *
* *
* Description : Initialise la classe générique des instructions. *
* *
* Retour : - *
* *
* Remarques : - *
* *
******************************************************************************/
static void g_arch_instruction_class_init(GArchInstructionClass *klass)
{
GObjectClass *object; /* Autre version de la classe */
object = G_OBJECT_CLASS(klass);
object->dispose = g_arch_instruction_dispose;
object->finalize = g_arch_instruction_finalize;
}
/******************************************************************************
* *
* Paramètres : iface = interface GLib à initialiser. *
* *
* Description : Procède à l'initialisation de l'interface de sérialisation. *
* *
* Retour : - *
* *
* Remarques : - *
* *
******************************************************************************/
static void g_arch_instruction_serializable_object_iface_init(GSerializableObjectInterface *iface)
{
iface->load = g_arch_instruction_load;
iface->store = g_arch_instruction_store;
}
/******************************************************************************
* *
* Paramètres : instr = instance à initialiser. *
* *
* Description : Initialise une instance d'instruction d'architecture. *
* *
* Retour : - *
* *
* Remarques : - *
* *
******************************************************************************/
static void g_arch_instruction_init(GArchInstruction *instr)
{
instr->rel_area = NULL;
instr->link_count = 0;
instr->links = NULL;
instr->operands = NULL;
}
/******************************************************************************
* *
* Paramètres : object = instance d'objet GLib à traiter. *
* *
* Description : Supprime toutes les références externes. *
* *
* Retour : - *
* *
* Remarques : - *
* *
******************************************************************************/
static void g_arch_instruction_dispose(GObject *object)
{
GArchInstruction *instr; /* Version spécialisée */
size_t count; /* Nombre d'opérandes en place */
size_t i; /* Boucle de parcours */
GArchOperand *op; /* Opérande à manipuler */
instr = G_ARCH_INSTRUCTION(object);
g_clear_object(&instr->rel_area);
g_arch_instruction_delete_all_links(instr);
g_thick_object_lock(G_THICK_OBJECT(instr));
count = g_arch_instruction_count_operands(instr);
for (i = 0; i < count; i++)
{
op = g_arch_instruction_get_operand(instr, 0);
rem_item_from_flat_array(&instr->operands, 0, sizeof(GArchOperand *));
/**
* Une fois pour l'obtention, une autre pour la libération !
*/
unref_object(op);
unref_object(op);
}
g_thick_object_unlock(G_THICK_OBJECT(instr));
G_OBJECT_CLASS(g_arch_instruction_parent_class)->dispose(G_OBJECT(instr));
}
/******************************************************************************
* *
* Paramètres : object = instance d'objet GLib à traiter. *
* *
* Description : Procède à la libération totale de la mémoire. *
* *
* Retour : - *
* *
* Remarques : - *
* *
******************************************************************************/
static void g_arch_instruction_finalize(GObject *object)
{
GArchInstruction *instr; /* Version spécialisée */
instr = G_ARCH_INSTRUCTION(object);
if (instr->links != NULL)
free(instr->links);
G_OBJECT_CLASS(g_arch_instruction_parent_class)->finalize(G_OBJECT(instr));
}
/******************************************************************************
* *
* Paramètres : instr = instance à initialiser pleinement. *
* tid = identifiant associé au type d'instructions ciblé. *
* *
* Description : Met en place une instruction d'architecture. *
* *
* Retour : Bilan de l'opération. *
* *
* Remarques : - *
* *
******************************************************************************/
bool g_arch_instruction_create(GArchInstruction *instr, itid_t tid)
{
bool result; /* Bilan à retourner */
instruction_extra_data_t extra; /* Données insérées à modifier */
result = true;
extra = GET_ARCH_INSTR_EXTRA(instr);
extra.tid = tid;
SET_ARCH_INSTR_EXTRA(instr, &extra);
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction quelconque à consulter. *
* *
* Description : Fournit l'identifiant correspondant à un type d'instructions.*
* *
* Retour : Identifiant unique par type d'instruction et architecture. *
* *
* Remarques : - *
* *
******************************************************************************/
itid_t g_arch_instruction_get_type_id(const GArchInstruction *instr)
{
itid_t result; /* Numéro à retourner */
instruction_extra_data_t extra; /* Données insérées à modifier */
extra = GET_ARCH_INSTR_EXTRA(instr);
result = extra.tid;
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction quelconque à consulter. *
* *
* Description : Indique l'encodage d'une instruction de façon détaillée. *
* *
* Retour : Description humaine de l'encodage utilisé. *
* *
* Remarques : - *
* *
******************************************************************************/
char *g_arch_instruction_get_encoding(const GArchInstruction *instr)
{
char *result; /* Encodage à retourner */
GArchInstructionClass *class; /* Classe des instructions */
class = G_ARCH_INSTRUCTION_GET_CLASS(instr);
result = class->get_encoding(instr);
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction d'assemblage à consulter. *
* *
* Description : Fournit le nom humain de l'instruction manipulée. *
* *
* Retour : Mot clef de bas niveau. *
* *
* Remarques : - *
* *
******************************************************************************/
char *g_arch_instruction_get_keyword(const GArchInstruction *instr)
{
char *result; /* Etiquette à retourner */
GArchInstructionClass *class; /* Classe des instructions */
class = G_ARCH_INSTRUCTION_GET_CLASS(instr);
result = class->get_keyword(instr);
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction quelconque à compléter. *
* area = portion de binaire incluant l'instruction. *
* start = adresse virtuelle et/ou position physique. *
* length = taille de l'instruction. *
* *
* Description : Calcule la localisation d'une instruction. *
* *
* Retour : - *
* *
* Remarques : - *
* *
******************************************************************************/
void g_arch_instruction_compute_range(GArchInstruction *instr, GBinaryPortion *area, const vmpa2t *start, phys_t length)
{
const mrange_t *a_range; /* Couverture de la portion */
phys_t diff; /* Décalage à appliquer */
a_range = g_binary_portion_get_range(area);
assert(mrange_contains_addr(a_range, start));
diff = compute_vmpa_diff(get_mrange_addr(a_range), start);
instr->rel_area = area;
ref_object(area);
init_rel_mrange(&instr->rel_range, diff, length);
}
/******************************************************************************
* *
* Paramètres : instr = instruction quelconque à consulter. *
* range = localisation de l'instruction. [OUT] *
* *
* Description : Fournit la place mémoire d'une instruction. *
* *
* Retour : Validité de la localisation : existence d'une définition ? *
* *
* Remarques : - *
* *
******************************************************************************/
bool g_arch_instruction_get_range(const GArchInstruction *instr, mrange_t *range)
{
bool result; /* Statut à retourner */
const mrange_t *a_range; /* Couverture de la portion */
vmpa2t start; /* Position de départ complète */
result = (instr->rel_area != NULL);
if (result)
{
a_range = g_binary_portion_get_range(instr->rel_area);
copy_vmpa(&start, get_mrange_addr(a_range));
advance_vmpa(&start, get_rel_mrange_offset(&instr->rel_range));
init_mrange(range, &start, get_rel_mrange_length(&instr->rel_range));
}
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction à venir modifier. *
* flag = drapeau d'information complémentaire à planter. *
* *
* Description : Ajoute une information complémentaire à une instruction. *
* *
* Retour : Bilan de l'opération. *
* *
* Remarques : - *
* *
******************************************************************************/
bool g_arch_instruction_set_flag(GArchInstruction *instr, ArchInstructionFlag flag)
{
bool result; /* Bilan à retourner */
instruction_extra_data_t extra; /* Données insérées à modifier */
assert(flag <= AIF_HIGH_USER);
extra = GET_ARCH_INSTR_EXTRA(instr);
result = !(extra.flags & flag);
extra.flags |= flag;
SET_ARCH_INSTR_EXTRA(instr, &extra);
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction à venir modifier. *
* flag = drapeau d'information complémentaire à planter. *
* *
* Description : Retire une information complémentaire à une instruction. *
* *
* Retour : Bilan de l'opération. *
* *
* Remarques : - *
* *
******************************************************************************/
bool g_arch_instruction_unset_flag(GArchInstruction *instr, ArchInstructionFlag flag)
{
bool result; /* Bilan à retourner */
instruction_extra_data_t extra; /* Données insérées à modifier */
assert(flag <= AIF_HIGH_USER);
extra = GET_ARCH_INSTR_EXTRA(instr);
result = (extra.flags & flag);
extra.flags &= ~flag;
SET_ARCH_INSTR_EXTRA(instr, &extra);
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction à venir consulter. *
* flag = drapeau d'information à rechercher. *
* *
* Description : Détermine si une instruction possède un fanion particulier. *
* *
* Retour : Bilan de la détection. *
* *
* Remarques : - *
* *
******************************************************************************/
bool g_arch_instruction_has_flag(const GArchInstruction *instr, ArchInstructionFlag flag)
{
bool result; /* Bilan à retourner */
instruction_extra_data_t extra; /* Données insérées à modifier */
assert(flag <= AIF_HIGH_USER);
extra = GET_ARCH_INSTR_EXTRA(instr);
result = (extra.flags & flag);
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction à venir consulter. *
* *
* Description : Fournit les particularités de l'instruction. *
* *
* Retour : Somme de tous les fanions associés à l'opérande. *
* *
* Remarques : - *
* *
******************************************************************************/
ArchInstructionFlag g_arch_instruction_get_flags(const GArchInstruction *instr)
{
ArchInstructionFlag result; /* Fanions à retourner */
instruction_extra_data_t extra; /* Données insérées à modifier */
extra = GET_ARCH_INSTR_EXTRA(instr);
result = extra.flags;
return result;
}
/* ---------------------------------------------------------------------------------- */
/* DEFINITION DES LIAISONS ENTRE INSTRUCTIONS */
/* ---------------------------------------------------------------------------------- */
/******************************************************************************
* *
* Paramètres : instr = instruction dont les liens sont à consulter. *
* dir = direction du lien recherché. *
* type = type de lien à détecter. *
* *
* Description : Détermine si un type de lien existe dans une instruction. *
* *
* Retour : Bilan du statut courant de l'instruction. *
* *
* Remarques : - *
* *
******************************************************************************/
static bool _g_arch_instruction_has_link(const GArchInstruction *instr, compact_ins_link_t dir, InstructionLinkType type)
{
bool result; /* Bilan à retourner */
uint16_t i; /* Boucle de parcours */
assert(g_thick_object_check_lock(G_THICK_OBJECT(instr)));
result = false;
for (i = 0; i < instr->link_count && !result; i++)
result = COMPACT_INS_LINK_DIR(instr->links[i]) == dir;
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction dont les liens sont à consulter. *
* type = type de lien à détecter. *
* *
* Description : Détermine si un type de lien amène à une instruction. *
* *
* Retour : Bilan du statut courant de l'instruction. *
* *
* Remarques : - *
* *
******************************************************************************/
bool g_arch_instruction_has_src_link(const GArchInstruction *instr, InstructionLinkType type)
{
bool result; /* Bilan à retourner */
result = _g_arch_instruction_has_link(instr, COMPACT_INS_LINK_FROM, type);
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction dont les liens sont à consulter. *
* type = type de lien à détecter. *
* *
* Description : Détermine si un type de lien émerge d'une instruction. *
* *
* Retour : Bilan du statut courant de l'instruction. *
* *
* Remarques : - *
* *
******************************************************************************/
bool g_arch_instruction_has_dest_link(const GArchInstruction *instr, InstructionLinkType type)
{
bool result; /* Bilan à retourner */
result = _g_arch_instruction_has_link(instr, COMPACT_INS_LINK_TO, type);
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction dont les liens sont à consulter. *
* dir = direction du lien recherché. *
* linked = seconde instruction à considérer. *
* *
* Description : Détermine si un lien existe entre deux instructions. *
* *
* Retour : Bilan du statut courant de l'instruction. *
* *
* Remarques : - *
* *
******************************************************************************/
static bool _g_arch_instruction_has_link_with(const GArchInstruction *instr, compact_ins_link_t dir, const GArchInstruction *linked)
{
bool result; /* Bilan à retourner */
uint16_t i; /* Boucle de parcours */
assert(g_thick_object_check_lock(G_THICK_OBJECT(instr)));
result = false;
for (i = 0; i < instr->link_count && !result; i++)
result = COMPACT_INS_LINK_PTR(instr->links[i]) == linked;
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction dont les liens sont à consulter. *
* src = seconde instruction à considérer. *
* *
* Description : Détermine si une instruction est source d'une autre. *
* *
* Retour : Bilan du statut courant de l'instruction. *
* *
* Remarques : - *
* *
******************************************************************************/
bool g_arch_instruction_has_src_link_with(const GArchInstruction *instr, const GArchInstruction *src)
{
bool result; /* Bilan à retourner */
result = _g_arch_instruction_has_link_with(instr, COMPACT_INS_LINK_FROM, src);
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction dont les liens sont à consulter. *
* dest = seconde instruction à considérer. *
* *
* Description : Détermine si une instruction est destination d'une autre. *
* *
* Retour : Bilan du statut courant de l'instruction. *
* *
* Remarques : - *
* *
******************************************************************************/
bool g_arch_instruction_has_dest_link_with(const GArchInstruction *instr, const GArchInstruction *dest)
{
bool result; /* Bilan à retourner */
result = _g_arch_instruction_has_link_with(instr, COMPACT_INS_LINK_TO, dest);
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction dont les informations sont à manipuler. *
* dest = ligne visée par la liaison (côté destination). *
* type = type de lien à construire. *
* *
* Description : Etablit un lien entre deux instructions. *
* *
* Retour : - *
* *
* Remarques : - *
* *
******************************************************************************/
void g_arch_instruction_link(GArchInstruction *instr, GArchInstruction *dest, InstructionLinkType type)
{
compact_ins_link_t new_from; /* Nouvel enregistrement #1 */
compact_ins_link_t new_to; /* Nouvel enregistrement #2 */
new_from = MAKE_COMPACT_INS_LINK(COMPACT_INS_LINK_FROM, instr, type);
ref_object(instr);
new_to = MAKE_COMPACT_INS_LINK(COMPACT_INS_LINK_TO, dest, type);
ref_object(dest);
g_thick_object_lock(G_THICK_OBJECT(instr));
g_thick_object_lock(G_THICK_OBJECT(dest));
dest->links = realloc(dest->links, ++dest->link_count * sizeof(compact_ins_link_t));
dest->links[dest->link_count - 1] = new_from;
instr->links = realloc(instr->links, ++instr->link_count * sizeof(compact_ins_link_t));
instr->links[instr->link_count - 1] = new_to;
g_thick_object_unlock(G_THICK_OBJECT(dest));
g_thick_object_unlock(G_THICK_OBJECT(instr));
}
/******************************************************************************
* *
* Paramètres : instr = instruction dont les informations sont à manipuler. *
* dest = ligne visée par la liaison (côté destination). *
* type = type de lien à construire. *
* *
* Description : Supprime un lien entre deux instructions. *
* *
* Retour : - *
* *
* Remarques : - *
* *
******************************************************************************/
void g_arch_instruction_unlink(GArchInstruction *instr, GArchInstruction *dest, InstructionLinkType type)
{
compact_ins_link_t old_from; /* Ancien enregistrement #1 */
compact_ins_link_t old_to; /* Ancien enregistrement #2 */
uint16_t i_from; /* Boucle de parcours #1 */
uint16_t i_to; /* Boucle de parcours #2 */
bool status; /* Bilan des recherches */
old_from = MAKE_COMPACT_INS_LINK(COMPACT_INS_LINK_FROM, instr, type);
old_to = MAKE_COMPACT_INS_LINK(COMPACT_INS_LINK_TO, dest, type);
g_thick_object_lock(G_THICK_OBJECT(instr));
g_thick_object_lock(G_THICK_OBJECT(dest));
for (i_from = 0; i_from < dest->link_count; i_from++)
if (dest->links[i_from] == old_from)
break;
for (i_to = 0; i_to < instr->link_count; i_to++)
if (instr->links[i_to] == old_to)
break;
assert((i_from < dest->link_count && i_to < instr->link_count)
|| (i_from == dest->link_count && i_to == instr->link_count));
status = (i_from < dest->link_count && i_to < instr->link_count);
if (status)
{
if ((i_from + 1) < dest->link_count)
memmove(&dest->links[i_from], &dest->links[i_from + 1],
(dest->link_count - i_from - 1) * sizeof(compact_ins_link_t));
dest->links = realloc(dest->links, --dest->link_count * sizeof(compact_ins_link_t));
if ((i_to + 1) < instr->link_count)
memmove(&instr->links[i_to], &instr->links[i_to + 1],
(instr->link_count - i_to - 1) * sizeof(compact_ins_link_t));
instr->links = realloc(instr->links, --instr->link_count * sizeof(compact_ins_link_t));
}
g_thick_object_unlock(G_THICK_OBJECT(dest));
g_thick_object_unlock(G_THICK_OBJECT(instr));
}
/******************************************************************************
* *
* Paramètres : instr = instruction dont les informations sont à manipuler. *
* dest = ligne visée par la liaison (côté destination). *
* old = ancien type de lien construit. *
* new = nouveau type de lien à construire. *
* *
* Description : Change la nature d'un lien entre deux instructions. *
* *
* Retour : true pour une mise à jour réussie, false sinon. *
* *
* Remarques : Le verrou doit être posé sur les destinations de 'instr'. *
* *
******************************************************************************/
bool g_arch_instruction_change_link(GArchInstruction *instr, GArchInstruction *dest, InstructionLinkType old, InstructionLinkType new)
{
bool result; /* Bilan à retourner */
compact_ins_link_t old_from; /* Ancien enregistrement #1 */
compact_ins_link_t new_from; /* Nouvel enregistrement #1 */
compact_ins_link_t old_to; /* Ancien enregistrement #2 */
compact_ins_link_t new_to; /* Nouvel enregistrement #2 */
uint16_t i_from; /* Boucle de parcours #1 */
uint16_t i_to; /* Boucle de parcours #2 */
old_from = MAKE_COMPACT_INS_LINK(COMPACT_INS_LINK_FROM, instr, old);
new_from = MAKE_COMPACT_INS_LINK(COMPACT_INS_LINK_FROM, instr, new);
old_to = MAKE_COMPACT_INS_LINK(COMPACT_INS_LINK_TO, dest, old);
new_to = MAKE_COMPACT_INS_LINK(COMPACT_INS_LINK_TO, dest, new);
g_thick_object_lock(G_THICK_OBJECT(instr));
g_thick_object_lock(G_THICK_OBJECT(dest));
for (i_from = 0; i_from < dest->link_count; i_from++)
if (dest->links[i_from] == old_from)
break;
for (i_to = 0; i_to < instr->link_count; i_to++)
if (instr->links[i_to] == old_to)
break;
assert((i_from < dest->link_count && i_to < instr->link_count)
|| (i_from == dest->link_count && i_to == instr->link_count));
result = (i_from < dest->link_count && i_to < instr->link_count);
if (result)
{
dest->links[i_from] = new_from;
instr->links[i_to] = new_to;
}
g_thick_object_unlock(G_THICK_OBJECT(dest));
g_thick_object_unlock(G_THICK_OBJECT(instr));
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction dont les informations sont à traiter. *
* *
* Description : Supprime tous les liens établis avec d'autres instructions. *
* *
* Retour : - *
* *
* Remarques : - *
* *
******************************************************************************/
void g_arch_instruction_delete_all_links(GArchInstruction *instr)
{
GArchInstruction *linked; /* Autre instruction liée */
InstructionLinkType type; /* Type de liaison */
g_thick_object_lock(G_THICK_OBJECT(instr));
while (g_arch_instruction_count_src_links(instr) > 0)
{
linked = g_arch_instruction_get_linked_source(instr, 0, &type);
g_thick_object_unlock(G_THICK_OBJECT(instr));
g_arch_instruction_unlink(linked, instr, type);
g_thick_object_lock(G_THICK_OBJECT(instr));
unref_object(linked);
}
while (g_arch_instruction_count_dest_links(instr) > 0)
{
linked = g_arch_instruction_get_linked_destination(instr, 0, &type);
g_thick_object_unlock(G_THICK_OBJECT(instr));
g_arch_instruction_unlink(instr, linked, type);
g_thick_object_lock(G_THICK_OBJECT(instr));
unref_object(linked);
}
g_thick_object_unlock(G_THICK_OBJECT(instr));
}
/******************************************************************************
* *
* Paramètres : instr = instruction dont les liens sont à consulter. *
* dir = direction des liens à considérer. *
* *
* Description : Fournit la quantité d'instructions pointant vers une autre. *
* *
* Retour : Nombre de ces liens. *
* *
* Remarques : - *
* *
******************************************************************************/
static size_t _g_arch_instruction_count_links(const GArchInstruction *instr, compact_ins_link_t dir)
{
size_t result; /* Nombre de liens à renvoyer */
uint16_t i; /* Boucle de parcours */
assert(g_thick_object_check_lock(G_THICK_OBJECT(instr)));
result = 0;
for (i = 0; i < instr->link_count; i++)
if (COMPACT_INS_LINK_DIR(instr->links[i]) == dir)
result++;
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction dont les liens sont à consulter. *
* *
* Description : Fournit la quantité d'instructions placées en source. *
* *
* Retour : Nombre de ces liens. *
* *
* Remarques : - *
* *
******************************************************************************/
size_t g_arch_instruction_count_src_links(const GArchInstruction *instr)
{
size_t result; /* Nombre de liens à renvoyer */
result = _g_arch_instruction_count_links(instr, COMPACT_INS_LINK_FROM);
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction dont les liens sont à consulter. *
* *
* Description : Fournit la quantité d'instructions placées en destination. *
* *
* Retour : Nombre de ces liens. *
* *
* Remarques : - *
* *
******************************************************************************/
size_t g_arch_instruction_count_dest_links(const GArchInstruction *instr)
{
size_t result; /* Nombre de liens à renvoyer */
result = _g_arch_instruction_count_links(instr, COMPACT_INS_LINK_TO);
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction dont les liens sont à consulter. *
* index = indice de l'élément à retrouver. *
* dir = direction des liens à considérer. *
* type = type de lien enregistré. [OUT] *
* *
* Description : Fournit les détails d'un lien donné avec une instruction. *
* *
* Retour : Autre instruction pointée par l'instruction, voire NULL. *
* *
* Remarques : - *
* *
******************************************************************************/
static GArchInstruction *_g_arch_instruction_get_linked_instruction(const GArchInstruction *instr, size_t index, compact_ins_link_t dir, InstructionLinkType *type)
{
GArchInstruction *result; /* Instance ciblée à renvoyer */
uint16_t i; /* Boucle de parcours */
assert(g_thick_object_check_lock(G_THICK_OBJECT(instr)));
result = NULL;
*type = ILT_COUNT;
for (i = 0; i < instr->link_count; i++)
{
if (COMPACT_INS_LINK_DIR(instr->links[i]) != dir)
continue;
if (index == 0)
{
result = COMPACT_INS_LINK_PTR(instr->links[i]);
*type = COMPACT_INS_LINK_TYPE(instr->links[i]);
}
else
index--;
}
if (result != NULL)
ref_object(result);
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction dont les liens sont à consulter. *
* index = indice de l'élément à retrouver. *
* type = type de lien enregistré. [OUT] *
* *
* Description : Fournit les détails d'une source donnée d'une instruction. *
* *
* Retour : Autre instruction pointée par l'instruction, voire NULL. *
* *
* Remarques : - *
* *
******************************************************************************/
GArchInstruction *g_arch_instruction_get_linked_source(const GArchInstruction *instr, size_t index, InstructionLinkType *type)
{
GArchInstruction *result; /* Instance ciblée à renvoyer */
result = _g_arch_instruction_get_linked_instruction(instr, index, COMPACT_INS_LINK_FROM, type);
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instruction dont les liens sont à consulter. *
* index = indice de l'élément à retrouver. *
* type = type de lien enregistré. [OUT] *
* *
* Description : Fournit les détails d'une destination donnée d'une instruct. *
* *
* Retour : Autre instruction pointée par l'instruction, voire NULL. *
* *
* Remarques : - *
* *
******************************************************************************/
GArchInstruction *g_arch_instruction_get_linked_destination(const GArchInstruction *instr, size_t index, InstructionLinkType *type)
{
GArchInstruction *result; /* Instance ciblée à renvoyer */
result = _g_arch_instruction_get_linked_instruction(instr, index, COMPACT_INS_LINK_TO, type);
return result;
}
/* ---------------------------------------------------------------------------------- */
/* MANIPULATION DES OPERANDES */
/* ---------------------------------------------------------------------------------- */
/******************************************************************************
* *
* Paramètres : instr = instance à consulter. *
* *
* Description : Indique la quantité d'opérandes présents dans l'instruction. *
* *
* Retour : Nombre d'opérandes attachés. *
* *
* Remarques : - *
* *
******************************************************************************/
size_t g_arch_instruction_count_operands(const GArchInstruction *instr)
{
size_t result; /* Décompte à retourner */
assert(g_thick_object_check_lock(G_THICK_OBJECT(instr)));
result = count_flat_array_items(instr->operands);
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instance à mettre à jour. *
* operand = instruction à venir associer. *
* *
* Description : Attache un opérande supplémentaire à une instruction. *
* *
* Retour : - *
* *
* Remarques : - *
* *
******************************************************************************/
void g_arch_instruction_attach_operand(GArchInstruction *instr, GArchOperand *operand)
{
GSingletonFactory *factory; /* Unise à instances uniques */
GSingletonCandidate *singleton; /* Instance retenue */
GArchOperand *stored; /* Forme d'opérande conservée */
assert(g_thick_object_check_lock(G_THICK_OBJECT(instr)));
factory = get_operands_factory();
singleton = g_singleton_factory_get_instance(factory, G_SINGLETON_CANDIDATE(operand));
unref_object(factory);
stored = G_ARCH_OPERAND(singleton);
add_item_to_flat_array(&instr->operands, &stored, sizeof(GArchOperand *));
}
/******************************************************************************
* *
* Paramètres : instr = instance à mettre à jour. *
* old = ancienne opérande à détacher. *
* new = nouvelle opérande à attacher. *
* *
* Description : Remplace un opérande d'une instruction par un autre. *
* *
* Retour : Bilan de l'opération. *
* *
* Remarques : - *
* *
******************************************************************************/
bool g_arch_instruction_replace_operand(GArchInstruction *instr, GArchOperand *old, GArchOperand *new)
{
bool result; /* Bilan à retourner */
size_t count; /* Nombre d'opérandes en place */
size_t i; /* Boucle de parcours */
GArchOperand *op; /* Opérande à manipuler */
GSingletonFactory *factory; /* Unise à instances uniques */
GSingletonCandidate *singleton; /* Instance retenue */
GArchOperand *stored; /* Forme d'opérande conservée */
assert(g_thick_object_check_lock(G_THICK_OBJECT(instr)));
result = false;
count = g_arch_instruction_count_operands(instr);
for (i = 0; i < count && !result; i++)
{
op = g_arch_instruction_get_operand(instr, i);
result = (op == old);
unref_object(op);
}
if (result)
{
factory = get_operands_factory();
singleton = g_singleton_factory_get_instance(factory, G_SINGLETON_CANDIDATE(new));
unref_object(factory);
stored = G_ARCH_OPERAND(singleton);
rpl_item_in_flat_array(instr->operands, i - 1, &stored, sizeof(GArchOperand *));
unref_object(old);
}
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instance à mettre à jour. *
* target = instruction à venir dissocier. *
* *
* Description : Détache un opérande liée d'une instruction. *
* *
* Retour : Bilan de l'opération. *
* *
* Remarques : - *
* *
******************************************************************************/
bool g_arch_instruction_detach_operand(GArchInstruction *instr, GArchOperand *target)
{
bool result; /* Bilan à retourner */
size_t count; /* Nombre d'opérandes en place */
size_t i; /* Boucle de parcours */
GArchOperand *op; /* Opérande à manipuler */
assert(g_thick_object_check_lock(G_THICK_OBJECT(instr)));
result = false;
count = g_arch_instruction_count_operands(instr);
for (i = 0; i < count && !result; i++)
{
op = g_arch_instruction_get_operand(instr, i);
result = (op == target);
unref_object(op);
}
if (result)
{
rem_item_from_flat_array(&instr->operands, i - 1, sizeof(GArchOperand *));
unref_object(target);
}
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instance à consulter. *
* index = indice de l'opérande concerné. *
* *
* Description : Fournit un opérande donné d'une instruction. *
* *
* Retour : Opérande trouvée. *
* *
* Remarques : - *
* *
******************************************************************************/
GArchOperand *g_arch_instruction_get_operand(const GArchInstruction *instr, size_t index)
{
GArchOperand *result; /* Opérande à retourner */
GArchOperand **ptr; /* Adresse dans le tableau */
assert(g_thick_object_check_lock(G_THICK_OBJECT(instr)));
ptr = get_flat_array_item(instr->operands, index, sizeof(GArchOperand *));
result = *ptr;
ref_object(result);
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instance à consulter. *
* target = instruction à venir retrouver. *
* *
* Description : Détermine le chemin conduisant à un opérande. *
* *
* Retour : Chemin d'accès à l'opérande ou NULL en cas d'absence. *
* *
* Remarques : - *
* *
******************************************************************************/
char *g_arch_instruction_find_operand_path(GArchInstruction *instr, const GArchOperand *target)
{
char *result; /* Chemin à retourner */
size_t count; /* Nombre d'opérandes en place */
size_t i; /* Boucle de parcours */
GArchOperand *op; /* Opérande à manipuler */
int ret; /* Bilan d'une construction */
char *sub_path; /* Sous-chemin emprunté */
result = NULL;
g_thick_object_lock(G_THICK_OBJECT(instr));
count = g_arch_instruction_count_operands(instr);
/* Première passe : accès direct */
for (i = 0; i < count && result == NULL; i++)
{
op = g_arch_instruction_get_operand(instr, i);
if (op == target)
{
ret = asprintf(&result, "%zu", i);
if (ret == -1)
{
LOG_ERROR_N("asprintf");
result = NULL;
}
}
unref_object(op);
}
/* Seconde passe : accès profond */
for (i = 0; i < count && result == NULL; i++)
{
op = g_arch_instruction_get_operand(instr, i);
sub_path = NULL;//g_arch_operand_find_inner_operand_path(op, target);
if (sub_path != NULL)
{
ret = asprintf(&result, "%zu:%s", i, sub_path);
if (ret == -1)
{
LOG_ERROR_N("asprintf");
result = NULL;
}
free(sub_path);
}
unref_object(op);
}
g_thick_object_unlock(G_THICK_OBJECT(instr));
return result;
}
/******************************************************************************
* *
* Paramètres : instr = instance à consulter. *
* path = chemin d'accès à un opérande à retrouver. *
* *
* Description : Obtient l'opérande correspondant à un chemin donné. *
* *
* Retour : Opérande trouvé ou NULL en cas d'échec. *
* *
* Remarques : - *
* *
******************************************************************************/
GArchOperand *g_arch_instruction_get_operand_from_path(GArchInstruction *instr, const char *path)
{
GArchOperand *result; /* Opérande trouvée à renvoyer */
size_t index; /* Indice de l'opérande visé */
char *end; /* Poursuite du parcours ? */
GArchOperand *found; /* Opérande trouvé */
result = NULL;
g_thick_object_lock(G_THICK_OBJECT(instr));
/* Recherche au premier niveau */
index = strtoul(path, &end, 10);
if ((index == ULONG_MAX && errno == ERANGE) || (index == 0 && errno == EINVAL))
{
LOG_ERROR_N("strtoul");
goto done;
}
found = g_arch_instruction_get_operand(instr, index);
if (found == NULL) goto done;
if (*end == '\0')
{
result = found;
goto done;
}
/* Recherche en profondeur */
assert(*end == ':');
result = NULL;//g_arch_operand_get_inner_operand_from_path(found, end + 1);
unref_object(found);
done:
g_thick_object_unlock(G_THICK_OBJECT(instr));
return result;
}
/* ---------------------------------------------------------------------------------- */
/* MECANISMES DE CONSERVATION ET RESTAURATION */
/* ---------------------------------------------------------------------------------- */
/******************************************************************************
* *
* Paramètres : object = élément GLib à constuire. *
* storage = conservateur de données à manipuler. *
* fd = flux ouvert en lecture. *
* *
* Description : Charge un objet depuis un flux de données. *
* *
* Retour : Bilan de l'opération. *
* *
* Remarques : - *
* *
******************************************************************************/
static bool g_arch_instruction_load(GSerializableObject *object, GObjectStorage *storage, int fd)
{
bool result; /* Bilan à retourner */
uleb128_t extra; /* Données embarquées */
/* Propriétés internes */
result = load_uleb128(&extra, fd);
if (result)
g_thick_object_set_extra(G_THICK_OBJECT(object), extra);
/* Liaisons avec d'autres instructions */
return result;
}
/******************************************************************************
* *
* Paramètres : object = élément GLib à consulter. *
* storage = conservateur de données à manipuler. *
* fd = flux ouvert en écriture. *
* *
* Description : Sauvegarde un objet dans un flux de données. *
* *
* Retour : Bilan de l'opération. *
* *
* Remarques : - *
* *
******************************************************************************/
static bool g_arch_instruction_store(const GSerializableObject *object, GObjectStorage *storage, int fd)
{
bool result; /* Bilan à retourner */
GArchInstruction *instr; /* Version spécialisée */
size_t src_count; /* Quantité de sources */
size_t dest_count; /* Quantité de destinations */
off64_t *ins_offsets; /* Emplacements d'instructions */
size_t i; /* Boucle de parcours */
GArchInstruction *linked; /* Instruction liée */
size_t op_count; /* Quantité d'opérandes */
off64_t *op_offsets; /* Emplacements d'opérandes */
GArchOperand *op; /* Opérande à traiter */
guint extra; /* Données embarquées */
InstructionLinkType type; /* Type de lien */
assert(g_thick_object_check_lock(G_THICK_OBJECT(object)));
/* Préparation des références aux instructions liées */
instr = G_ARCH_INSTRUCTION(object);
src_count = g_arch_instruction_count_src_links(instr);
dest_count = g_arch_instruction_count_dest_links(instr);
ins_offsets = malloc((src_count + dest_count) * sizeof(off64_t));
for (i = 0; i < src_count && result; i++)
{
linked = g_arch_instruction_get_linked_source(instr, i, (InstructionLinkType []) { 0 });
result = g_object_storage_store_object(storage, "instructions",
G_SERIALIZABLE_OBJECT(linked), &ins_offsets[i]);
unref_object(linked);
}
for (i = 0; i < dest_count && result; i++)
{
linked = g_arch_instruction_get_linked_destination(instr, i, (InstructionLinkType []) { 0 });
result = g_object_storage_store_object(storage, "instructions",
G_SERIALIZABLE_OBJECT(linked), &ins_offsets[src_count + i]);
unref_object(linked);
}
if (!result)
goto exit_with_ins_off;
/* Préparation des références aux opérandes embarqués */
op_count = g_arch_instruction_count_operands(instr);
op_offsets = malloc(op_count * sizeof(off64_t));
for (i = 0; i < op_count && result; i++)
{
op = g_arch_instruction_get_operand(instr, i);
result = g_object_storage_store_object(storage, "operandss",
G_SERIALIZABLE_OBJECT(op), &op_offsets[i]);
unref_object(op);
}
if (!result)
goto exit_with_op_off;
/* Propriétés internes */
extra = g_thick_object_get_extra(G_THICK_OBJECT(object));
result = store_uleb128((uleb128_t []) { extra }, fd);
if (!result) goto exit;
/* Liaisons avec d'autres instructions */
instr = G_ARCH_INSTRUCTION(object);
src_count = g_arch_instruction_count_src_links(instr);
dest_count = g_arch_instruction_count_dest_links(instr);
result = store_uleb128((uleb128_t []) { src_count }, fd);
if (!result) goto exit;
result = store_uleb128((uleb128_t []) { dest_count }, fd);
if (!result) goto exit;
for (i = 0; i < src_count && result; i++)
{
linked = g_arch_instruction_get_linked_source(instr, i, &type);
result = store_uleb128((uleb128_t []) { type }, fd);
unref_object(linked);
if (result)
result = store_uleb128((uleb128_t []) { ins_offsets[i] }, fd);
}
for (i = 0; i < dest_count && result; i++)
{
linked = g_arch_instruction_get_linked_destination(instr, i, &type);
result = store_uleb128((uleb128_t []) { type }, fd);
unref_object(linked);
if (result)
result = store_uleb128((uleb128_t []) { ins_offsets[src_count + i] }, fd);
}
/* Opérandes embarqués */
result = store_uleb128((uleb128_t []) { op_count }, fd);
if (!result) goto exit;
for (i = 0; i < op_count && result; i++)
result = store_uleb128((uleb128_t []) { op_offsets[i] }, fd);
exit:
exit_with_op_off:
free(op_offsets);
exit_with_ins_off:
free(ins_offsets);
return result;
}
#if 0
/******************************************************************************
* *
* Paramètres : instr = instruction quelconque à traiter. *
* type = type de procédure à utiliser. *
* proc = représentation de l'architecture utilisée. *
* context = contexte associé à la phase de désassemblage. *
* format = accès aux données du binaire d'origine. *
* *
* Description : Complète un désassemblage accompli pour une instruction. *
* *
* Retour : - *
* *
* Remarques : - *
* *
******************************************************************************/
void g_arch_instruction_call_hook(GArchInstruction *instr, InstrProcessHook type, GArchProcessor *proc, GProcContext *context, GExeFormat *format)
{
GArchInstructionClass *class; /* Classe des instructions */
class = G_ARCH_INSTRUCTION_GET_CLASS(instr);
if (class->call_hook != NULL)
class->call_hook(instr, type, proc, context, format);
}
#endif