/* 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