/* Chrysalide - Outil d'analyse de fichiers binaires
* instruction.h - prototypes pour la 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 .
*/
#ifndef _ARCH_INSTRUCTION_H
#define _ARCH_INSTRUCTION_H
#include
#include
#include "operand.h"
#include "vmpa.h"
#include "../glibext/helpers.h"
#include "../glibext/portion.h"
/* ----------------------- DEFINITION GENERIQUE D'INSTRUCTION ----------------------- */
#define G_TYPE_ARCH_INSTRUCTION (g_arch_instruction_get_type())
DECLARE_GTYPE(GArchInstruction, g_arch_instruction, G, ARCH_INSTRUCTION);
/* Type pour les types d'instructions */
typedef uint16_t itid_t;
/* Fournit l'identifiant correspondant à un type d'instructions. */
itid_t g_arch_instruction_get_type_id(const GArchInstruction *);
/* Indique l'encodage d'une instruction de façon détaillée. */
char *g_arch_instruction_get_encoding(const GArchInstruction *);
/* Fournit le nom humain de l'instruction manipulée. */
char *g_arch_instruction_get_keyword(const GArchInstruction *);
/* Type de masques pour les encodages d'instructions */
typedef enum _InstructionBytesMask
{
/**
* Correspond aux bits fixes : pas de valeurs de registre ni de valeur entière.
*/
IBM_LOOSE,
/**
* Dissimulation des références à des éléments externes pouvant varier avec
* entre compilations : adresses de saut ou d'appel, références vers des tables,
* etc.
*/
IBM_LOCAL,
/**
* Dissimulation des déplacements à partir d'une base.
*/
IBM_STRICT,
/**
* Conservation de toutes les valeurs immédiates et dissimulation des registres.
*/
IBM_LARGE,
IBM_COUNT
} InstructionBytesMask;
/* Calcule la localisation d'une instruction. */
void g_arch_instruction_compute_range(GArchInstruction *, GBinaryPortion *, const vmpa2t *, phys_t);
/* Fournit la place mémoire d'une instruction. */
bool g_arch_instruction_get_range(const GArchInstruction *, mrange_t *);
#define AIF_USER_BIT 4
typedef enum _ArchInstructionFlag
{
AIF_NONE = (0 << 0), /* Aucune information */
AIF_ROUTINE_START = (1 << 0), /* Début de routine */
AIF_RETURN_POINT = (1 << 1), /* Retour de fonction appelée */
AIF_COND_RETURN_POINT = (1 << 2), /* Retour éventuel de fonction */
AIF_CALL = (1 << 3), /* Instruction d'appel */
AIF_LOW_USER = (1 << AIF_USER_BIT), /* Premier bit disponible */
AIF_HIGH_USER = (1 << 7), /* Dernier bit disponible */
} ArchInstructionFlag;
#define AIF_USER_FLAG(n) (1 << (AIF_USER_BIT + n))
/* Ajoute une information complémentaire à une instruction. */
bool g_arch_instruction_set_flag(GArchInstruction *, ArchInstructionFlag);
/* Retire une information complémentaire à une instruction. */
bool g_arch_instruction_unset_flag(GArchInstruction *, ArchInstructionFlag);
/* Détermine si une instruction possède un fanion particulier. */
bool g_arch_instruction_has_flag(const GArchInstruction *, ArchInstructionFlag);
/* Fournit les particularités de l'instruction. */
ArchInstructionFlag g_arch_instruction_get_flags(const GArchInstruction *);
/* Types de crochet de traitement */
typedef enum _InstrProcessHook
{
IPH_FETCH, /* Itinéraire de désassemblage */
IPH_LINK, /* Edition des liens */
IPH_POST, /* Résolution des symboles */
IPH_COUNT
} InstrProcessHook;
/* ------------------- DEFINITION DES LIAISONS ENTRE INSTRUCTIONS ------------------- */
/* Typage des instructions rencontrées */
typedef enum _InstructionLinkType
{
ILT_EXEC_FLOW, /* Raccord attendu entre blocs */
ILT_JUMP, /* Saut inconditionnel */
ILT_CASE_JUMP, /* Saut suite à aiguillage */
ILT_JUMP_IF_TRUE, /* Saut conditionnel (si vrai) */
ILT_JUMP_IF_FALSE, /* Saut conditionnel (si faux) */
ILT_LOOP, /* Retour en arrière (boucle) */
ILT_CALL, /* Appel d'une fonction */
ILT_CATCH_EXCEPTION, /* Gestion d'une exception */
ILT_REF, /* Simple référence croisée */
ILT_COUNT
} InstructionLinkType;
/* Détermine si un type de lien amène à une instruction. */
bool g_arch_instruction_has_src_link(const GArchInstruction *, InstructionLinkType);
/* Détermine si un type de lien émerge d'une instruction. */
bool g_arch_instruction_has_dest_link(const GArchInstruction *, InstructionLinkType);
/* Détermine si une instruction est source d'une autre. */
bool g_arch_instruction_has_src_link_with(const GArchInstruction *, const GArchInstruction *);
/* Détermine si une instruction est destination d'une autre. */
bool g_arch_instruction_has_dest_link_with(const GArchInstruction *, const GArchInstruction *);
/* Etablit un lien entre deux instructions. */
void g_arch_instruction_link(GArchInstruction *, GArchInstruction *, InstructionLinkType);
/* Supprime un lien entre deux instructions. */
void g_arch_instruction_unlink(GArchInstruction *, GArchInstruction *, InstructionLinkType);
/* Change la nature d'un lien entre deux instructions. */
bool g_arch_instruction_change_link(GArchInstruction *, GArchInstruction *, InstructionLinkType, InstructionLinkType);
/* Supprime tous les liens établis avec d'autres instructions. */
void g_arch_instruction_delete_all_links(GArchInstruction *);
/* Fournit la quantité d'instructions placées en source. */
size_t g_arch_instruction_count_src_links(const GArchInstruction *);
/* Fournit la quantité d'instructions placées en destination. */
size_t g_arch_instruction_count_dest_links(const GArchInstruction *);
/* Fournit les détails d'une source donnée d'une instruction. */
GArchInstruction *g_arch_instruction_get_linked_source(const GArchInstruction *, size_t, InstructionLinkType *);
/* Fournit les détails d'une destination donnée d'une instruction. */
GArchInstruction *g_arch_instruction_get_linked_destination(const GArchInstruction *, size_t, InstructionLinkType *);
/* --------------------------- MANIPULATION DES OPERANDES --------------------------- */
/* Indique la quantité d'opérandes présents dans l'instruction. */
size_t g_arch_instruction_count_operands(const GArchInstruction *);
/* Attache un opérande supplémentaire à une instruction. */
void g_arch_instruction_attach_operand(GArchInstruction *, GArchOperand *);
/* Remplace un opérande d'une instruction par un autre. */
bool g_arch_instruction_replace_operand(GArchInstruction *, GArchOperand *, GArchOperand *);
/* Détache un opérande liée d'une instruction. */
bool g_arch_instruction_detach_operand(GArchInstruction *, GArchOperand *);
/* Fournit un opérande donné d'une instruction. */
GArchOperand *g_arch_instruction_get_operand(const GArchInstruction *, size_t);
/* Détermine le chemin conduisant à un opérande. */
char *g_arch_instruction_find_operand_path(GArchInstruction *, const GArchOperand *);
/* Obtient l'opérande correspondant à un chemin donné. */
GArchOperand *g_arch_instruction_get_operand_from_path(GArchInstruction *, const char *);
#if 0
/**
* La définition de "GArchProcessor", utile aux traitements complémentaires, ne peut
* se faire en incluant le fichier d'en-tête "processor.h", pour cause de références
* circulaires.
*
* On procède donc à une seconde déclaration, en attendant éventuellement mieux.
*/
/* Depuis "processeur.h" : définition générique d'un processeur d'architecture (instance) */
typedef struct _GArchProcessor GArchProcessor;
/* Complète un désassemblage accompli pour une instruction. */
typedef void (* instr_hook_fc) (GArchInstruction *, GArchProcessor *, GProcContext *, GExeFormat *);
/* Complète un désassemblage accompli pour une instruction. */
void g_arch_instruction_call_hook(GArchInstruction *, InstrProcessHook, GArchProcessor *, GProcContext *, GExeFormat *);
#endif
#endif /* _ARCH_INSTRUCTION_H */