diff options
| author | Cyrille Bagard <nocbos@gmail.com> | 2025-04-01 00:05:16 (GMT) | 
|---|---|---|
| committer | Cyrille Bagard <nocbos@gmail.com> | 2025-04-01 00:05:16 (GMT) | 
| commit | bb7e4c5e6e4c51da0d9b1a33b571b0c64851c1a8 (patch) | |
| tree | 4575210322bf6838f538a4f58967c0a2a0d9cabc /src/arch/instructions | |
| parent | 70ed4dc99c75c13797b41164959c753ffbc4572b (diff) | |
Restore most features of core instructions.
Diffstat (limited to 'src/arch/instructions')
| -rw-r--r-- | src/arch/instructions/Makefile.am | 11 | ||||
| -rw-r--r-- | src/arch/instructions/raw-int.h | 56 | ||||
| -rw-r--r-- | src/arch/instructions/raw-ui.c | 261 | ||||
| -rw-r--r-- | src/arch/instructions/raw-ui.h | 37 | ||||
| -rw-r--r-- | src/arch/instructions/raw.c | 696 | ||||
| -rw-r--r-- | src/arch/instructions/raw.h | 41 | ||||
| -rw-r--r-- | src/arch/instructions/undefined-int.h | 52 | ||||
| -rw-r--r-- | src/arch/instructions/undefined-ui.c | 102 | ||||
| -rw-r--r-- | src/arch/instructions/undefined-ui.h | 37 | ||||
| -rw-r--r-- | src/arch/instructions/undefined.c | 383 | ||||
| -rw-r--r-- | src/arch/instructions/undefined.h | 28 | 
11 files changed, 793 insertions, 911 deletions
| diff --git a/src/arch/instructions/Makefile.am b/src/arch/instructions/Makefile.am index 28cf90f..d6fc4bd 100644 --- a/src/arch/instructions/Makefile.am +++ b/src/arch/instructions/Makefile.am @@ -1,12 +1,19 @@ -noinst_LTLIBRARIES = libarchinstructions.la +noinst_LTLIBRARIES = libarchinstructions.la libarchinstructionsui.la  libarchinstructions_la_SOURCES =			\ +	raw-int.h								\  	raw.h raw.c								\  	undefined-int.h							\  	undefined.h undefined.c -libarchinstructions_la_CFLAGS = $(TOOLKIT_CFLAGS) $(LIBXML_CFLAGS) +libarchinstructions_la_CFLAGS = $(LIBGOBJ_CFLAGS) + +libarchinstructionsui_la_SOURCES =			\ +	raw-ui.h raw-ui.c						\ +	undefined-ui.h undefined-ui.c + +libarchinstructionsui_la_CFLAGS = $(LIBGTK4_CFLAGS)  devdir = $(includedir)/chrysalide/$(subdir:src/%=core/%) diff --git a/src/arch/instructions/raw-int.h b/src/arch/instructions/raw-int.h new file mode 100644 index 0000000..4a5e64b --- /dev/null +++ b/src/arch/instructions/raw-int.h @@ -0,0 +1,56 @@ + +/* Chrysalide - Outil d'analyse de fichiers binaires + * raw-int.h - prototypes pour la définition interne des instructions de données brutes + * + * Copyright (C) 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 <http://www.gnu.org/licenses/>. + */ + + +#ifndef _ARCH_INSTRUCTIONS_RAW_INT_H +#define _ARCH_INSTRUCTIONS_RAW_INT_H + + +#include "raw.h" +#include "../instruction-int.h" + + + +/* Définition générique d'une instruction brute d'architecture (instance) */ +struct _GRawInstruction +{ +    GArchInstruction parent;                /* A laisser en premier        */ + +}; + +/* Définition générique d'une instruction brute d'architecture (instance) */ +struct _GRawInstructionClass +{ +    GArchInstructionClass parent;           /* A laisser en premier        */ + +}; + + +/* Met en place une instruction de type 'db/dw/etc' simple. */ +bool g_raw_instruction_create_value(GRawInstruction *, GBinaryPortion *, const vmpa2t *, MemoryDataSize, uint64_t); + +/* Met en place une instruction de type 'db/dw/etc' étendue. */ +bool g_raw_instruction_create_array(GRawInstruction *, GBinaryPortion *, vmpa2t *, MemoryDataSize, const GBinContent *, size_t, SourceEndian); + + + +#endif  /* _ARCH_INSTRUCTIONS_RAW_INT_H */ diff --git a/src/arch/instructions/raw-ui.c b/src/arch/instructions/raw-ui.c new file mode 100644 index 0000000..1026dfb --- /dev/null +++ b/src/arch/instructions/raw-ui.c @@ -0,0 +1,261 @@ + +/* Chrysalide - Outil d'analyse de fichiers binaires + * raw-ui.c - opérandes représentant des instructions de données brutes sous forme graphique + * + * Copyright (C) 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 <http://www.gnu.org/licenses/>. + */ + + +#include "raw-ui.h" + + +#include <assert.h> +#include <ctype.h> +#include <malloc.h> + + +#include "raw.h" +#include "../operand-ui.h" +#include "../operands/immediate.h" +#include "../../glibext/objhole.h" +#include "../../glibext/options/asm.h" + + + +/* Etablit dans une ligne de rendu le contenu représenté. */ +static void g_raw_instruction_ui_populate_line(const GTokenGenerator *, size_t, size_t, GBufferLine *, void *); + + + +/****************************************************************************** +*                                                                             * +*  Paramètres  : iface = interface GLib à initialiser.                        * +*                                                                             * +*  Description : Procède à l'initialisation de l'interface de génération.     * +*                                                                             * +*  Retour      : -                                                            * +*                                                                             * +*  Remarques   : -                                                            * +*                                                                             * +******************************************************************************/ + +void g_raw_instruction_ui_token_generator_iface_init(GTokenGeneratorInterface *iface) +{ +    iface->populate = g_raw_instruction_ui_populate_line; + +} + + +/****************************************************************************** +*                                                                             * +*  Paramètres  : generator = générateur à utiliser pour l'impression.         * +*                index     = indice de cette même ligne dans le tampon global.* +*                repeat    = indice d'utilisations successives du générateur. * +*                line      = ligne de rendu à compléter.                      * +*                data      = éventuelle donnée complémentaire fournie.        * +*                                                                             * +*  Description : Etablit dans une ligne de rendu le contenu représenté.       * +*                                                                             * +*  Retour      : -                                                            * +*                                                                             * +*  Remarques   : -                                                            * +*                                                                             * +******************************************************************************/ + +static void g_raw_instruction_ui_populate_line(const GTokenGenerator *generator, size_t index, size_t repeat, GBufferLine *line, void *data) +{ +    GArchInstruction *instr;                /* Version spécialisée #1      */ +    GRawInstruction *raw;                   /* Version spécialisée #2      */ +    GBinContent *content;                   /* Contenu brut d'origine      */ +    mrange_t range;                         /* Emplacement couvert         */ +    phys_t max_displayed_len;               /* Quantité de code affichée   */ +    char *key;                              /* Mot clef principal          */ +    char *string;                           /* Chaîne reconstituée         */ +    size_t iter;                            /* Tête d'écriture             */ +    bool first;                             /* Mémorise une énumération    */ +    size_t count;                           /* Nombre d'opérandes en place */ +    size_t i;                               /* Boucle de parcours          */ +    GArchOperand *op;                       /* Opérande à manipuler        */ +    GImmediateOperand *imm;                 /* Version opérande de valeur  */ +    char byte;                              /* Octet à afficher (ou pas)   */ +#ifndef NDEBUG +    bool status;                            /* Bilan d'une récupération    */ +#endif + +    instr = G_ARCH_INSTRUCTION(generator); +    raw = G_RAW_INSTRUCTION(instr); +    content = G_BIN_CONTENT(data); + +    g_thick_object_lock(G_THICK_OBJECT(instr)); + +    /* Prologue */ + +    if (g_arch_instruction_get_range(instr, &range)) +    { +        /* Localisation */ + +        g_buffer_line_fill_physical(line, ACO_PHYSICAL, MDS_32_BITS_UNSIGNED, get_mrange_addr(&range)); + +        g_buffer_line_fill_virtual(line, ACO_VIRTUAL, MDS_32_BITS_UNSIGNED, get_mrange_addr(&range)); + +        /* Contenu */ + +        if (g_raw_instruction_is_padding(raw)) +            max_displayed_len = 0; + +        else if (g_raw_instruction_is_string(raw)) +            max_displayed_len = 1; + +        else +        { +            max_displayed_len = get_mrange_length(&range); +            max_displayed_len /= g_arch_instruction_count_operands(instr); +        } + +        g_buffer_line_fill_content(line, ACO_BINARY, content, &range, max_displayed_len); + +    } + +    /* Instruction proprement dite */ + +    key = g_arch_instruction_get_keyword(instr); + +    g_buffer_line_append_text(line, ACO_ASSEMBLY_HEAD, TRT_INSTRUCTION, SL(key), NULL, G_OBJECT(instr)); + +    free(key); + +    /* Contenu sous forme d'opérandes */ + +    if (g_raw_instruction_is_padding(raw)) +        g_buffer_line_append_text(line, ACO_ASSEMBLY, TRT_CHR_PRINTABLE, STCSL("..."), NULL, NULL); + +    else +    { +        string = NULL; +        iter = 0; + +        first = true; + +        count = g_arch_instruction_count_operands(instr); + +        for (i = 0; i < count; i++) +        { +            op = g_arch_instruction_get_operand(instr, i); + +            if (!G_IS_IMMEDIATE_OPERAND(op)) +                goto fallback; + +            imm = G_IMMEDIATE_OPERAND(op); + +            if (g_immediate_operand_get_size(imm) != MDS_8_BITS) +                goto fallback; + +            if (!g_raw_instruction_is_string(raw) && g_immediate_operand_get_display(imm) != IOD_CHAR) +                goto fallback; + +#ifndef NDEBUG +            status = g_immediate_operand_get_value(imm, MDS_8_BITS, &byte); +            assert(status); +#else +            g_immediate_operand_get_value(imm, MDS_8_BITS, &byte); +#endif + +            /* Si le caractère doit apparaître en hexadécimal... */ + +            if (!isprint(byte)) +                goto fallback; + +            /* Impression de l'octet */ + +            if (string == NULL) +            { +                string = calloc(count + 3, sizeof(char)); + +                strcpy(string, "\""); +                iter = 1; + +            } + +            string[iter++] = byte; + +            unref_object(op); + +            continue; + + fallback: + +            /* Si une chaîne précède */ + +            if (string != NULL && iter > 1) +            { +                if (!first) +                { +                    g_buffer_line_append_text(line, ACO_ASSEMBLY, TRT_PUNCT, STCSL(","), NULL, NULL); +                    g_buffer_line_append_text(line, ACO_ASSEMBLY, TRT_CHR_PRINTABLE, STCSL(" "), NULL, NULL); +                } +                else +                    first = false; + +                string[iter++] = '"'; + +                g_buffer_line_append_text(line, ACO_ASSEMBLY, TRT_STRING, string, iter, NULL, NULL); + +                iter = 1; + +            } + +            /* Intégration en tant qu'opérande classique */ + +            if (!first) +            { +                g_buffer_line_append_text(line, ACO_ASSEMBLY, TRT_PUNCT, STCSL(","), NULL, NULL); +                g_buffer_line_append_text(line, ACO_ASSEMBLY, TRT_CHR_PRINTABLE, STCSL(" "), NULL, NULL); +            } +            else +                first = false; + +            g_arch_operand_ui_print(G_ARCH_OPERAND_UI(op), line); + +            unref_object(op); + +        } + +        /* Si au final une chaîne traine encore */ + +        if (string != NULL && iter > 1) +        { +            if (!first) +            { +                g_buffer_line_append_text(line, ACO_ASSEMBLY, TRT_PUNCT, STCSL(","), NULL, NULL); +                g_buffer_line_append_text(line, ACO_ASSEMBLY, TRT_CHR_PRINTABLE, STCSL(" "), NULL, NULL); +            } + +            string[iter++] = '"'; + +            g_buffer_line_append_text(line, ACO_ASSEMBLY, TRT_STRING, string, iter, NULL, NULL); + +        } + +        if (string != NULL) +            free(string); + +    } + +    g_thick_object_unlock(G_THICK_OBJECT(instr)); + +} diff --git a/src/arch/instructions/raw-ui.h b/src/arch/instructions/raw-ui.h new file mode 100644 index 0000000..cd604e6 --- /dev/null +++ b/src/arch/instructions/raw-ui.h @@ -0,0 +1,37 @@ + +/* Chrysalide - Outil d'analyse de fichiers binaires + * raw-ui.h - prototypes pour les opérandes représentant des instructions de données brutes sous forme graphique + * + * Copyright (C) 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 <http://www.gnu.org/licenses/>. + */ + + +#ifndef _ARCH_OPERANDS_RAW_UI_H +#define _ARCH_OPERANDS_RAW_UI_H + + +#include "../../glibext/generator-int.h" + + + +/* Procède à l'initialisation de l'interface de génération. */ +void g_raw_instruction_ui_token_generator_iface_init(GTokenGeneratorInterface *); + + + +#endif  /* _ARCH_OPERANDS_RAW_UI_H */ diff --git a/src/arch/instructions/raw.c b/src/arch/instructions/raw.c index 26282fa..87297f1 100644 --- a/src/arch/instructions/raw.c +++ b/src/arch/instructions/raw.c @@ -2,7 +2,7 @@  /* Chrysalide - Outil d'analyse de fichiers binaires   * raw.c - instructions pures vues de l'esprit   * - * Copyright (C) 2014-2020 Cyrille Bagard + * Copyright (C) 2014-2025 Cyrille Bagard   *   *  This file is part of Chrysalide.   * @@ -25,36 +25,19 @@  #include <assert.h> -#include <ctype.h>  #include <string.h>  #include <i18n.h> -#include "../instruction-int.h" +#include "raw-int.h"  #include "../operands/immediate.h" -#include "../operands/target.h" -#include "../../core/columns.h" +//#include "../operands/target.h" // FIXME -/* ------------------------- INSTRUCTION INCONNUE / DONNEES ------------------------- */ - - -/* Définition générique d'une instruction brute d'architecture (instance) */ -struct _GRawInstruction -{ -    GArchInstruction parent;                /* A laisser en premier        */ - -}; - -/* Définition générique d'une instruction brute d'architecture (classe) */ -struct _GRawInstructionClass -{ -    GArchInstructionClass parent;           /* A laisser en premier        */ - -}; +/* --------------------- INSTRUCTION AVEC JEU DE DONNEES BRUTES --------------------- */  /* Initialise la classe des instructions brutes d'architecture. */ @@ -64,40 +47,26 @@ static void g_raw_instruction_class_init(GRawInstructionClass *);  static void g_raw_instruction_init(GRawInstruction *);  /* Supprime toutes les références externes. */ -static void g_raw_instruction_dispose(GRawInstruction *); +static void g_raw_instruction_dispose(GObject *);  /* Procède à la libération totale de la mémoire. */ -static void g_raw_instruction_finalize(GRawInstruction *); - -/* Indique l'encodage d'une instruction de façon détaillée. */ -static const char *g_raw_instruction_get_encoding(const GRawInstruction *); - -/* Fournit le nom humain de l'instruction manipulée. */ -static const char *g_raw_instruction_get_keyword(const GRawInstruction *); +static void g_raw_instruction_finalize(GObject *); -/* -------------------- CONSERVATION SUR DISQUE DES INSTRUCTIONS -------------------- */ +/* --------------------- IMPLEMENTATION DES FONCTIONS DE CLASSE --------------------- */ -/* Charge une instruction depuis une mémoire tampon. */ -static bool g_raw_instruction_unserialize(GRawInstruction *, GAsmStorage *, GBinFormat *, packed_buffer_t *); - -/* Sauvegarde une instruction dans une mémoire tampon. */ -static bool g_raw_instruction_serialize(GRawInstruction *, GAsmStorage *, packed_buffer_t *); - - - -/* ------------------------ OFFRE DE CAPACITES DE GENERATION ------------------------ */ - +/* Indique l'encodage d'une instruction de façon détaillée. */ +static char *g_raw_instruction_get_encoding(const GArchInstruction *); -/* Ajoute à un tampon GLib le contenu de l'instance spécifiée. */ -static void g_raw_instruction_print(GRawInstruction *, GBufferLine *, size_t, size_t, const GBinContent *); +/* Fournit le nom humain de l'instruction manipulée. */ +static char *g_raw_instruction_get_keyword(const GArchInstruction *);  /* ---------------------------------------------------------------------------------- */ -/*                           INSTRUCTION INCONNUE / DONNEES                           */ +/*                       INSTRUCTION AVEC JEU DE DONNEES BRUTES                       */  /* ---------------------------------------------------------------------------------- */ @@ -124,18 +93,13 @@ static void g_raw_instruction_class_init(GRawInstructionClass *klass)      object = G_OBJECT_CLASS(klass); -    object->dispose = (GObjectFinalizeFunc/* ! */)g_raw_instruction_dispose; -    object->finalize = (GObjectFinalizeFunc)g_raw_instruction_finalize; +    object->dispose = g_raw_instruction_dispose; +    object->finalize = g_raw_instruction_finalize;      instr = G_ARCH_INSTRUCTION_CLASS(klass); -    instr->get_encoding = (get_instruction_encoding_fc)g_raw_instruction_get_encoding; -    instr->get_keyword = (get_instruction_keyword_fc)g_raw_instruction_get_keyword; - -    instr->unserialize = (unserialize_instruction_fc)g_raw_instruction_unserialize; -    instr->serialize = (serialize_instruction_fc)g_raw_instruction_serialize; - -    instr->print = (print_instruction_fc)g_raw_instruction_print; +    instr->get_encoding = g_raw_instruction_get_encoding; +    instr->get_keyword = g_raw_instruction_get_keyword;  } @@ -160,7 +124,7 @@ static void g_raw_instruction_init(GRawInstruction *instr)  /******************************************************************************  *                                                                             * -*  Paramètres  : instr = instance d'objet GLib à traiter.                     * +*  Paramètres  : object = instance d'objet GLib à traiter.                    *  *                                                                             *  *  Description : Supprime toutes les références externes.                     *  *                                                                             * @@ -170,16 +134,16 @@ static void g_raw_instruction_init(GRawInstruction *instr)  *                                                                             *  ******************************************************************************/ -static void g_raw_instruction_dispose(GRawInstruction *instr) +static void g_raw_instruction_dispose(GObject *object)  { -    G_OBJECT_CLASS(g_raw_instruction_parent_class)->dispose(G_OBJECT(instr)); +    G_OBJECT_CLASS(g_raw_instruction_parent_class)->dispose(object);  }  /******************************************************************************  *                                                                             * -*  Paramètres  : instr = instance d'objet GLib à traiter.                     * +*  Paramètres  : object = instance d'objet GLib à traiter.                    *  *                                                                             *  *  Description : Procède à la libération totale de la mémoire.                *  *                                                                             * @@ -189,16 +153,17 @@ static void g_raw_instruction_dispose(GRawInstruction *instr)  *                                                                             *  ******************************************************************************/ -static void g_raw_instruction_finalize(GRawInstruction *instr) +static void g_raw_instruction_finalize(GObject *object)  { -    G_OBJECT_CLASS(g_raw_instruction_parent_class)->finalize(G_OBJECT(instr)); +    G_OBJECT_CLASS(g_raw_instruction_parent_class)->finalize(object);  }  /******************************************************************************  *                                                                             * -*  Paramètres  : addr  = position à associer à l'instruction.                 * +*  Paramètres  : area  = portion de binaire incluant l'instruction.           * +*                addr  = adresse virtuelle et/ou position physique.           *  *                size  = taille de l'opérande souhaitée.                      *  *                value = valeur sur x bits à venir récupérer.                 *  *                                                                             * @@ -210,127 +175,106 @@ static void g_raw_instruction_finalize(GRawInstruction *instr)  *                                                                             *  ******************************************************************************/ -GArchInstruction *g_raw_instruction_new_from_value(const vmpa2t *addr, MemoryDataSize size, uint64_t value) +GArchInstruction *g_raw_instruction_new_from_value(GBinaryPortion *area, const vmpa2t *addr, MemoryDataSize size, uint64_t value)  {      GArchInstruction *result;               /* Instruction à retourner     */ -    GArchOperand *operand;                  /* Octet non décodé à afficher */ -    mrange_t range;                         /* Couverture de l'instruction */      result = g_object_new(G_TYPE_RAW_INSTRUCTION, NULL); -    operand = g_imm_operand_new_from_value(size, value); -    if (operand == NULL) goto error; +    if (!g_raw_instruction_create_value(G_RAW_INSTRUCTION(result), area, addr, size, value)) +        g_clear_object(&result); + +    return result; + +} + + +/****************************************************************************** +*                                                                             * +*  Paramètres  : instr   = instance à initialiser pleinement.                 * +*                area    = portion de binaire incluant l'instruction.         * +*                addr    = adresse virtuelle et/ou position physique.         * +*                size    = taille de chacun des éléments à représenter.       * +*                value = valeur sur x bits à venir récupérer.                 * +*                                                                             * +*  Description : Met en place une instruction de type 'db/dw/etc' simple.     * +*                                                                             * +*  Retour      : Bilan de l'opération.                                        * +*                                                                             * +*  Remarques   : -                                                            * +*                                                                             * +******************************************************************************/ + +bool g_raw_instruction_create_value(GRawInstruction *instr, GBinaryPortion *area, const vmpa2t *addr, MemoryDataSize size, uint64_t value) +{ +    bool result;                            /* Bilan à retourner           */ +    GArchOperand *operand;                  /* Octet non décodé à afficher */ +    uint16_t length;                        /* Taille de l'instruction     */ + +    result = false; + +    operand = g_immediate_operand_new_from_value(size, value); +    if (operand == NULL) goto exit;      g_arch_operand_set_flag(operand, IOF_ZERO_PADDING_BY_DEFAULT | IOF_ZERO_PADDING); -    g_arch_instruction_attach_extra_operand(result, operand); +    g_thick_object_lock(G_THICK_OBJECT(instr)); + +    g_arch_instruction_attach_operand(G_ARCH_INSTRUCTION(instr), operand); +    unref_object(operand); + +    g_thick_object_unlock(G_THICK_OBJECT(instr));      switch (size)      {          case MDS_8_BITS_UNSIGNED:          case MDS_8_BITS_SIGNED: -            init_mrange(&range, addr, 1); +            length = 1;              break;          case MDS_16_BITS_UNSIGNED:          case MDS_16_BITS_SIGNED: -            init_mrange(&range, addr, 2); +            length = 2;              break;          case MDS_32_BITS_UNSIGNED:          case MDS_32_BITS_SIGNED: -            init_mrange(&range, addr, 4); +            length = 4;              break;          case MDS_64_BITS_UNSIGNED:          case MDS_64_BITS_SIGNED: -            init_mrange(&range, addr, 8); +            length = 8;              break;          default:              assert(false); -            goto error; +            goto exit;              break;      } -    g_arch_instruction_set_range(result, &range); +    g_arch_instruction_compute_range(G_ARCH_INSTRUCTION(instr), area, addr, length); -    return result; - - error: - -    g_object_unref(G_OBJECT(result)); - -    return NULL; - -} +    result = true; - -/****************************************************************************** -*                                                                             * -*  Paramètres  : content = flux de données à analyser.                        * -*                addr    = position courante dans ce flux. [OUT]              * -*                                                                             * -*  Description : Crée une instruction de type 'db/dw/etc' pour un uleb128.    * -*                                                                             * -*  Retour      : Instruction mise en place.                                   * -*                                                                             * -*  Remarques   : -                                                            * -*                                                                             * -******************************************************************************/ - -GArchInstruction *g_raw_instruction_new_uleb128(const GBinContent *content, vmpa2t *addr) -{ -    GArchInstruction *result;               /* Instruction à retourner     */ -    vmpa2t start;                           /* Départ original de lecture  */ -    uleb128_t value;                        /* Valeur uleb128 à représenter*/ -    phys_t diff;                            /* Couverture de la lecture    */ -    MemoryDataSize leb_size;                /* Taille de la valeur         */ -    GArchOperand *operand;                  /* Octet non décodé à afficher */ -    mrange_t range;                         /* Couverture de l'instruction */ - -    result = NULL; - -    copy_vmpa(&start, addr); - -    if (!g_binary_content_read_uleb128(content, addr, &value)) -        goto error; - -    diff = compute_vmpa_diff(&start, addr); - -    leb_size = MDS_FROM_BYTES(diff); -    assert(leb_size != MDS_UNDEFINED); - -    result = g_object_new(G_TYPE_RAW_INSTRUCTION, NULL); - -    init_mrange(&range, &start, diff); -    g_arch_instruction_set_range(result, &range); - -    operand = g_imm_operand_new_from_value(leb_size, (uint64_t)value); -    if (operand == NULL) goto error; - -    g_arch_operand_set_flag(operand, IOF_ZERO_PADDING_BY_DEFAULT | IOF_ZERO_PADDING); - -    g_arch_instruction_attach_extra_operand(result, operand); + exit:      return result; - error: - -    g_clear_object(&result); - -    return NULL; -  }  /******************************************************************************  *                                                                             * -*  Paramètres  : content = flux de données à analyser.                        * -*                addr    = position courante dans ce flux. [OUT]              * +*  Paramètres  : area    = portion de binaire incluant l'instruction.         * +*                addr    = adresse virtuelle et/ou position physique.         * +*                size    = taille de chacun des éléments à représenter.       * +*                content = flux de données à analyser.                        * +*                count   = nombre de ces éléments.                            * +*                endian  = ordre des bits dans la source.                     *  *                                                                             * -*  Description : Crée une instruction de type 'db/dw/etc' pour un sleb128.    * +*  Description : Crée une instruction de type 'db/dw/etc' étendue.            *  *                                                                             *  *  Retour      : Instruction mise en place.                                   *  *                                                                             * @@ -338,256 +282,174 @@ GArchInstruction *g_raw_instruction_new_uleb128(const GBinContent *content, vmpa  *                                                                             *  ******************************************************************************/ -GArchInstruction *g_raw_instruction_new_sleb128(const GBinContent *content, vmpa2t *addr) +GArchInstruction *g_raw_instruction_new_array(GBinaryPortion *area, vmpa2t *addr, MemoryDataSize size, const GBinContent *content, size_t count, SourceEndian endian)  {      GArchInstruction *result;               /* Instruction à retourner     */ -    vmpa2t start;                           /* Départ original de lecture  */ -    uleb128_t value;                        /* Valeur uleb128 à représenter*/ -    phys_t diff;                            /* Couverture de la lecture    */ -    MemoryDataSize leb_size;                /* Taille de la valeur         */ -    GArchOperand *operand;                  /* Octet non décodé à afficher */ -    mrange_t range;                         /* Couverture de l'instruction */ - -    result = NULL; - -    copy_vmpa(&start, addr); - -    if (!g_binary_content_read_uleb128(content, addr, &value)) -        goto error; - -    diff = compute_vmpa_diff(&start, addr); - -    leb_size = MDS_FROM_BYTES(diff) | MDS_SIGN; -    assert(leb_size != MDS_SIGN);      result = g_object_new(G_TYPE_RAW_INSTRUCTION, NULL); -    init_mrange(&range, &start, diff); -    g_arch_instruction_set_range(result, &range); - -    operand = g_imm_operand_new_from_value(leb_size, (uint64_t)value); -    if (operand == NULL) goto error; - -    g_arch_operand_set_flag(operand, IOF_ZERO_PADDING_BY_DEFAULT | IOF_ZERO_PADDING); - -    g_arch_instruction_attach_extra_operand(result, operand); +    if (!g_raw_instruction_create_array(G_RAW_INSTRUCTION(result), area, addr, size, content, count, endian)) +        g_clear_object(&result);      return result; - error: - -    g_clear_object(&result); - -    return NULL; -  }  /******************************************************************************  *                                                                             * -*  Paramètres  : content = flux de données à analyser.                        * +*  Paramètres  : instr   = instance à initialiser pleinement.                 * +*                area    = portion de binaire incluant l'instruction.         * +*                addr    = adresse virtuelle et/ou position physique.         *  *                size    = taille de chacun des éléments à représenter.       * +*                content = flux de données à analyser.                        *  *                count   = nombre de ces éléments.                            * -*                addr    = position courante dans ce flux. [OUT]              *  *                endian  = ordre des bits dans la source.                     *  *                                                                             * -*  Description : Crée une instruction de type 'db/dw/etc' étendue.            * +*  Description : Met en place une instruction de type 'db/dw/etc' étendue.    *  *                                                                             * -*  Retour      : Instruction mise en place.                                   * +*  Retour      : Bilan de l'opération.                                        *  *                                                                             *  *  Remarques   : -                                                            *  *                                                                             *  ******************************************************************************/ -GArchInstruction *g_raw_instruction_new_array(const GBinContent *content, MemoryDataSize size, size_t count, vmpa2t *addr, SourceEndian endian) +bool g_raw_instruction_create_array(GRawInstruction *instr, GBinaryPortion *area, vmpa2t *addr, MemoryDataSize size, const GBinContent *content, size_t count, SourceEndian endian)  { -    GArchInstruction *result;               /* Instruction à retourner     */ -    vmpa2t old;                             /* Sauvegarde de la position   */ +    bool result;                            /* Bilan à retourner           */ +    vmpa2t start;                           /* Sauvegarde de la position   */      size_t i;                               /* Boucle de parcours          */      GArchOperand *operand;                  /* Octet non décodé à afficher */ -    mrange_t range;                         /* Couverture de l'instruction */ +    phys_t diff;                            /* Décalage à appliquer        */ + +    result = false;      /* Par soucis de cohérence */ -    if (count == 0) return NULL; +    if (count == 0) +        goto exit; -    result = g_object_new(G_TYPE_RAW_INSTRUCTION, NULL); +    copy_vmpa(&start, addr); -    copy_vmpa(&old, addr); +    g_thick_object_lock(G_THICK_OBJECT(instr));      for (i = 0; i < count; i++)      { -        operand = g_imm_operand_new_from_data(size, content, addr, endian); -        if (operand == NULL) goto error; +        operand = g_immediate_operand_new_from_data(size, content, addr, (bool []){ false /* unused */ }, endian); +        if (operand == NULL) break;          g_arch_operand_set_flag(operand, IOF_ZERO_PADDING_BY_DEFAULT | IOF_ZERO_PADDING); -        g_arch_instruction_attach_extra_operand(result, operand); +        g_arch_instruction_attach_operand(G_ARCH_INSTRUCTION(instr), operand); +        unref_object(operand);      } -    init_mrange(&range, &old, compute_vmpa_diff(addr, &old)); +    g_thick_object_unlock(G_THICK_OBJECT(instr)); -    g_arch_instruction_set_range(result, &range); +    if (i < count) +        goto exit; -    return result; +    diff = compute_vmpa_diff(addr, &start); + +    g_arch_instruction_compute_range(G_ARCH_INSTRUCTION(instr), area, &start, diff); - error: +    result = true; -    g_object_unref(G_OBJECT(result)); + exit: -    return NULL; +    return result;  }  /******************************************************************************  *                                                                             * -*  Paramètres  : instr = instruction quelconque à consulter.                  * +*  Paramètres  : instr      = instruction à traiter.                          * +*                is_padding = nouveau statut à associer au contenu.           *  *                                                                             * -*  Description : Indique l'encodage d'une instruction de façon détaillée.     * +*  Description : Marque l'instruction comme ne contenant que du bourrage.     *  *                                                                             * -*  Retour      : Description humaine de l'encodage utilisé.                   * +*  Retour      : -                                                            *  *                                                                             *  *  Remarques   : -                                                            *  *                                                                             *  ******************************************************************************/ -static const char *g_raw_instruction_get_encoding(const GRawInstruction *instr) +void g_raw_instruction_mark_as_padding(GRawInstruction *instr, bool is_padding)  { -    const char *result;                     /* Description à retourner     */ - -    if (g_raw_instruction_is_string(instr)) -        result = _("String"); +    if (is_padding) +        g_arch_instruction_set_flag(G_ARCH_INSTRUCTION(instr), RIF_PADDING);      else -        result = _("Raw"); - -    return result; +        g_arch_instruction_unset_flag(G_ARCH_INSTRUCTION(instr), RIF_PADDING);  }  /******************************************************************************  *                                                                             * -*  Paramètres  : instr = instruction d'assemblage à consulter.                * +*  Paramètres  : instr      = instruction à traiter.                          * +*                is_padding = nouveau statut à associer au contenu.           *  *                                                                             * -*  Description : Fournit le nom humain de l'instruction manipulée.            * +*  Description : Indique si le contenu de l'instruction est du bourrage.      *  *                                                                             * -*  Retour      : Mot clef de bas niveau.                                      * +*  Retour      : Statut du contenu de l'instruction.                          *  *                                                                             *  *  Remarques   : -                                                            *  *                                                                             *  ******************************************************************************/ -static const char *g_raw_instruction_get_keyword(const GRawInstruction *instr) +bool g_raw_instruction_is_padding(const GRawInstruction *instr)  { -    GArchOperand *operand;                  /* Octet décodé à afficher     */ -    MemoryDataSize size;                    /* Taille de valeur associée   */ - -    static char *defines[] = { "dn", "db", "dw", "dd", "dq" }; - -    operand = g_arch_instruction_get_operand(G_ARCH_INSTRUCTION(instr), 0); - -    if (G_IS_TARGET_OPERAND(operand)) -        size = g_target_operand_get_size(G_TARGET_OPERAND(operand)); -    else -        size = g_imm_operand_get_size(G_IMM_OPERAND(operand)); +    bool result;                            /* Indication à retourner      */ -    g_object_unref(G_OBJECT(operand)); +    result = g_arch_instruction_has_flag(G_ARCH_INSTRUCTION(instr), RIF_PADDING); -    return defines[MDS_RANGE(size)]; +    return result;  } - -/* ---------------------------------------------------------------------------------- */ -/*                      CONSERVATION SUR DISQUE DES INSTRUCTIONS                      */ -/* ---------------------------------------------------------------------------------- */ - -  /******************************************************************************  *                                                                             * -*  Paramètres  : instr   = instruction d'assemblage à consulter.              * -*                storage = mécanisme de sauvegarde à manipuler.               * -*                format  = format binaire chargé associé à l'architecture.    * -*                pbuf    = zone tampon à remplir.                             * +*  Paramètres  : instr     = instruction à traiter.                           * +*                is_string = nouveau statut à associer au contenu.            *  *                                                                             * -*  Description : Charge une instruction depuis une mémoire tampon.            * +*  Description : Marque l'instruction comme contenant une chaîne de texte.    *  *                                                                             * -*  Retour      : Bilan de l'opération.                                        * +*  Retour      : -                                                            *  *                                                                             *  *  Remarques   : -                                                            *  *                                                                             *  ******************************************************************************/ -static bool g_raw_instruction_unserialize(GRawInstruction *instr, GAsmStorage *storage, GBinFormat *format, packed_buffer_t *pbuf) +void g_raw_instruction_mark_as_string(GRawInstruction *instr, bool is_string)  { -    bool result;                            /* Bilan à retourner           */ -    GArchInstructionClass *parent;          /* Classe parente à consulter  */ -    uint8_t boolean;                        /* Valeur booléenne            */ - -    parent = G_ARCH_INSTRUCTION_CLASS(g_raw_instruction_parent_class); - -    result = parent->unserialize(G_ARCH_INSTRUCTION(instr), storage, format, pbuf); - -    if (result) -    { -        result = extract_packed_buffer(pbuf, &boolean, sizeof(uint8_t), false); - -        if (result) -            g_raw_instruction_mark_as_padding(instr, (boolean == 1)); - -    } - -    if (result) -    { -        result = extract_packed_buffer(pbuf, &boolean, sizeof(uint8_t), false); - -        if (result) -            g_raw_instruction_mark_as_string(instr, (boolean == 1)); - -    } - -    return result; +    if (is_string) +        g_arch_instruction_set_flag(G_ARCH_INSTRUCTION(instr), RIF_STRING); +    else +        g_arch_instruction_unset_flag(G_ARCH_INSTRUCTION(instr), RIF_STRING);  }  /******************************************************************************  *                                                                             * -*  Paramètres  : instr   = instruction d'assemblage à consulter.              * -*                storage = mécanisme de sauvegarde à manipuler.               * -*                pbuf    = zone tampon à remplir.                             * +*  Paramètres  : instr     = instruction à traiter.                           * +*                is_string = nouveau statut à associer au contenu.            *  *                                                                             * -*  Description : Sauvegarde une instruction dans une mémoire tampon.          * +*  Description : Indique si le contenu de l'instruction est un texte.         *  *                                                                             * -*  Retour      : Bilan de l'opération.                                        * +*  Retour      : Statut du contenu de l'instruction.                          *  *                                                                             *  *  Remarques   : -                                                            *  *                                                                             *  ******************************************************************************/ -static bool g_raw_instruction_serialize(GRawInstruction *instr, GAsmStorage *storage, packed_buffer_t *pbuf) +bool g_raw_instruction_is_string(const GRawInstruction *instr)  { -    bool result;                            /* Bilan à retourner           */ -    GArchInstructionClass *parent;          /* Classe parente à consulter  */ -    uint8_t boolean;                        /* Valeur booléenne            */ - -    parent = G_ARCH_INSTRUCTION_CLASS(g_raw_instruction_parent_class); - -    result = parent->serialize(G_ARCH_INSTRUCTION(instr), storage, pbuf); - -    if (result) -    { -        boolean = (g_raw_instruction_is_padding(instr) ? 1 : 0); -        result = extend_packed_buffer(pbuf, &boolean, sizeof(uint8_t), false); -    } +    bool result;                            /* Indication à retourner      */ -    if (result) -    { -        boolean = (g_raw_instruction_is_string(instr) ? 1 : 0); -        result = extend_packed_buffer(pbuf, &boolean, sizeof(uint8_t), false); -    } +    result = g_arch_instruction_has_flag(G_ARCH_INSTRUCTION(instr), RIF_STRING);      return result; @@ -596,285 +458,73 @@ static bool g_raw_instruction_serialize(GRawInstruction *instr, GAsmStorage *sto  /* ---------------------------------------------------------------------------------- */ -/*                          OFFRE DE CAPACITES DE GENERATION                          */ +/*                       IMPLEMENTATION DES FONCTIONS DE CLASSE                       */  /* ---------------------------------------------------------------------------------- */  /******************************************************************************  *                                                                             * -*  Paramètres  : instr   = instruction d'assemblage à représenter.            * -*                line    = ligne de rendu à compléter.                        * -*                index   = indice de cette même ligne dans le tampon global.  * -*                repeat  = indice d'utilisations successives du générateur.   * -*                content = éventuel contenu binaire brut à imprimer.          * +*  Paramètres  : instr = instruction quelconque à consulter.                  *  *                                                                             * -*  Description : Ajoute à un tampon GLib le contenu de l'instance spécifiée.  * +*  Description : Indique l'encodage d'une instruction de façon détaillée.     *  *                                                                             * -*  Retour      : -                                                            * +*  Retour      : Description humaine de l'encodage utilisé.                   *  *                                                                             *  *  Remarques   : -                                                            *  *                                                                             *  ******************************************************************************/ -static void g_raw_instruction_print(GRawInstruction *instr, GBufferLine *line, size_t index, size_t repeat, const GBinContent *content) +static char *g_raw_instruction_get_encoding(const GArchInstruction *instr)  { -    GArchInstruction *base;                 /* Autre version de l'instance */ -    phys_t max_displayed_len;               /* Quantité de code affichée   */ -    const char *key;                        /* Mot clef principal          */ -    size_t klen;                            /* Taille de ce mot clef       */ -    char *string;                           /* Chaîne reconstituée         */ -    size_t iter;                            /* Tête d'écriture             */ -    bool first;                             /* Mémorise une énumération    */ -    size_t count;                           /* Nombre d'opérandes en place */ -    size_t i;                               /* Boucle de parcours          */ -    GArchOperand *op;                       /* Opérande à manipuler        */ -    GImmOperand *imm;                       /* Version opérande de valeur  */ -    char byte;                              /* Octet à afficher (ou pas)   */ -#ifndef NDEBUG -    bool status;                            /* Bilan d'une récupération    */ -#endif - -    base = G_ARCH_INSTRUCTION(instr); - -    /* Localisation */ - -    g_buffer_line_fill_phys(line, DLC_PHYSICAL, MDS_32_BITS_UNSIGNED, get_mrange_addr(&base->range)); - -    g_buffer_line_fill_virt(line, DLC_VIRTUAL, MDS_32_BITS_UNSIGNED, get_mrange_addr(&base->range)); - -    /* Contenu */ - -    if (g_raw_instruction_is_padding(instr)) -        max_displayed_len = 0; - -    else if (g_raw_instruction_is_string(instr)) -        max_displayed_len = 1; - -    else -    { -        max_displayed_len = get_mrange_length(&base->range); -        max_displayed_len /= g_arch_instruction_count_operands(base); -    } - -    g_buffer_line_fill_content(line, DLC_BINARY, content, &base->range, max_displayed_len); - -    /* Zone du code d'assemblage */ +    char *result;                           /* Description à retourner     */ +    GRawInstruction *raw;                   /* Version spécialisée         */ -    key = g_arch_instruction_get_keyword(base); -    klen = strlen(key); - -    g_buffer_line_append_text(line, DLC_ASSEMBLY_HEAD, key, klen, RTT_INSTRUCTION, NULL); - -    if (g_raw_instruction_is_padding(instr)) -        g_buffer_line_append_text(line, DLC_ASSEMBLY, "...", 3, RTT_RAW, NULL); +    raw = G_RAW_INSTRUCTION(instr); +    if (g_raw_instruction_is_string(raw)) +        result = strdup(_("String"));      else -    { -        string = NULL; -        iter = 0; - -        first = true; - -        g_arch_instruction_lock_operands(base); - -        count = _g_arch_instruction_count_operands(base); - -        for (i = 0; i < count; i++) -        { -            op = _g_arch_instruction_get_operand(base, i); - -            if (!G_IS_IMM_OPERAND(op)) -                goto grip_fallback; - -            imm = G_IMM_OPERAND(op); - -            if (g_imm_operand_get_size(imm) != MDS_8_BITS) -                goto grip_fallback; - -            if (!g_raw_instruction_is_string(instr) && g_imm_operand_get_display(imm) != IOD_CHAR) -                goto grip_fallback; - -#ifndef NDEBUG -            status = g_imm_operand_get_value(imm, MDS_8_BITS, &byte); -            assert(status); -#else -            g_imm_operand_get_value(imm, MDS_8_BITS, &byte); -#endif - -            /* Si le caractère doit apparaître en hexadécimal... */ - -            if (!isprint(byte)) -                goto grip_fallback; - -            /* Impression de l'octet */ - -            if (string == NULL) -            { -                string = (char *)calloc(count + 3, sizeof(char)); - -                strcpy(string, "\""); -                iter = 1; - -            } - -            string[iter++] = byte; - -            g_object_unref(G_OBJECT(op)); - -            continue; - - grip_fallback: - -            /* Si une chaîne précède */ - -            if (string != NULL && iter > 1) -            { -                if (!first) -                { -                    g_buffer_line_append_text(line, DLC_ASSEMBLY, ",", 1, RTT_PUNCT, NULL); -                    g_buffer_line_append_text(line, DLC_ASSEMBLY, " ", 1, RTT_RAW, NULL); -                } -                else -                    first = false; - -                string[iter++] = '"'; - -                g_buffer_line_append_text(line, DLC_ASSEMBLY, string, iter, RTT_STRING, NULL); - -                iter = 1; - -            } - -            /* Intégration en tant qu'opérande classique */ - -            if (!first) -            { -                g_buffer_line_append_text(line, DLC_ASSEMBLY, ",", 1, RTT_PUNCT, NULL); -                g_buffer_line_append_text(line, DLC_ASSEMBLY, " ", 1, RTT_RAW, NULL); -            } -            else -                first = false; - -            g_arch_operand_print(op, line); - -            g_object_unref(G_OBJECT(op)); - -        } - -        /* Si au final une chaîne traine encore */ - -        if (string != NULL && iter > 1) -        { -            if (!first) -            { -                g_buffer_line_append_text(line, DLC_ASSEMBLY, ",", 1, RTT_PUNCT, NULL); -                g_buffer_line_append_text(line, DLC_ASSEMBLY, " ", 1, RTT_RAW, NULL); -            } - -            string[iter++] = '"'; - -            g_buffer_line_append_text(line, DLC_ASSEMBLY, string, iter, RTT_STRING, NULL); - -        } +        result = strdup(_("Raw")); -        g_arch_instruction_unlock_operands(base); - -        if (string != NULL) -            free(string); - -    } - -} - - -/****************************************************************************** -*                                                                             * -*  Paramètres  : instr      = instruction à traiter.                          * -*                is_padding = nouveau statut à associer au contenu.           * -*                                                                             * -*  Description : Marque l'instruction comme ne contenant que du bourrage.     * -*                                                                             * -*  Retour      : -                                                            * -*                                                                             * -*  Remarques   : -                                                            * -*                                                                             * -******************************************************************************/ - -void g_raw_instruction_mark_as_padding(GRawInstruction *instr, bool is_padding) -{ -    if (is_padding) -        g_arch_instruction_set_flag(G_ARCH_INSTRUCTION(instr), RIF_PADDING); -    else -        g_arch_instruction_unset_flag(G_ARCH_INSTRUCTION(instr), RIF_PADDING); +    return result;  }  /******************************************************************************  *                                                                             * -*  Paramètres  : instr      = instruction à traiter.                          * -*                is_padding = nouveau statut à associer au contenu.           * +*  Paramètres  : instr = instruction d'assemblage à consulter.                *  *                                                                             * -*  Description : Indique si le contenu de l'instruction est du bourrage.      * +*  Description : Fournit le nom humain de l'instruction manipulée.            *  *                                                                             * -*  Retour      : Statut du contenu de l'instruction.                          * +*  Retour      : Mot clef de bas niveau.                                      *  *                                                                             *  *  Remarques   : -                                                            *  *                                                                             *  ******************************************************************************/ -bool g_raw_instruction_is_padding(const GRawInstruction *instr) +static char *g_raw_instruction_get_keyword(const GArchInstruction *instr)  { -    bool result;                            /* Indication à retourner      */ - -    result = g_arch_instruction_has_flag(G_ARCH_INSTRUCTION(instr), RIF_PADDING); - -    return result; - -} +    char *result;                           /* Désignation à retourner     */ +    GArchOperand *operand;                  /* Octet décodé à afficher     */ +    MemoryDataSize size;                    /* Taille de valeur associée   */ +    static char *defines[] = { "dn", "db", "dw", "dd", "dq" }; -/****************************************************************************** -*                                                                             * -*  Paramètres  : instr     = instruction à traiter.                           * -*                is_string = nouveau statut à associer au contenu.            * -*                                                                             * -*  Description : Marque l'instruction comme contenant une chaîne de texte.    * -*                                                                             * -*  Retour      : -                                                            * -*                                                                             * -*  Remarques   : -                                                            * -*                                                                             * -******************************************************************************/ +    g_thick_object_lock(G_THICK_OBJECT(instr)); -void g_raw_instruction_mark_as_string(GRawInstruction *instr, bool is_string) -{ -    if (is_string) -        g_arch_instruction_set_flag(G_ARCH_INSTRUCTION(instr), RIF_STRING); -    else -        g_arch_instruction_unset_flag(G_ARCH_INSTRUCTION(instr), RIF_STRING); - -} +    operand = g_arch_instruction_get_operand(instr, 0); +    /*if (G_IS_TARGET_OPERAND(operand)) FIXME  +        size = g_target_operand_get_size(G_TARGET_OPERAND(operand)); +        else*/ +        size = g_immediate_operand_get_size(G_IMMEDIATE_OPERAND(operand)); -/****************************************************************************** -*                                                                             * -*  Paramètres  : instr     = instruction à traiter.                           * -*                is_string = nouveau statut à associer au contenu.            * -*                                                                             * -*  Description : Indique si le contenu de l'instruction est un texte.         * -*                                                                             * -*  Retour      : Statut du contenu de l'instruction.                          * -*                                                                             * -*  Remarques   : -                                                            * -*                                                                             * -******************************************************************************/ +    unref_object(operand); -bool g_raw_instruction_is_string(const GRawInstruction *instr) -{ -    bool result;                            /* Indication à retourner      */ +    g_thick_object_unlock(G_THICK_OBJECT(instr)); -    result = g_arch_instruction_has_flag(G_ARCH_INSTRUCTION(instr), RIF_STRING); +    result = strdup(defines[MDS_RANGE(size)]);      return result; diff --git a/src/arch/instructions/raw.h b/src/arch/instructions/raw.h index 4e92cd4..712f877 100644 --- a/src/arch/instructions/raw.h +++ b/src/arch/instructions/raw.h @@ -1,8 +1,8 @@  /* Chrysalide - Outil d'analyse de fichiers binaires - * raw.h - prototypes pour les instructions pures vues de l'esprit + * raw.h - prototypes pour les instructions de données brutes   * - * Copyright (C) 2014-2020 Cyrille Bagard + * Copyright (C) 2014-2025 Cyrille Bagard   *   *  This file is part of Chrysalide.   * @@ -25,46 +25,24 @@  #define _ARCH_INSTRUCTIONS_RAW_H -#include <glib-object.h> - -  #include "../instruction.h"  #include "../vmpa.h" +#include "../../analysis/content.h" +#include "../../glibext/helpers.h" -/* ------------------------- INSTRUCTION INCONNUE / DONNEES ------------------------- */ - - -#define G_TYPE_RAW_INSTRUCTION            g_raw_instruction_get_type() -#define G_RAW_INSTRUCTION(obj)            (G_TYPE_CHECK_INSTANCE_CAST((obj), G_TYPE_RAW_INSTRUCTION, GRawInstruction)) -#define G_IS_RAW_INSTRUCTION(obj)         (G_TYPE_CHECK_INSTANCE_TYPE((obj), G_TYPE_RAW_INSTRUCTION)) -#define G_RAW_INSTRUCTION_CLASS(klass)    (G_TYPE_CHECK_CLASS_CAST((klass), G_TYPE_RAW_INSTRUCTION, GRawInstructionClass)) -#define G_IS_RAW_INSTRUCTION_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE((klass), G_TYPE_RAW_INSTRUCTION)) -#define G_RAW_INSTRUCTION_GET_CLASS(obj)  (G_TYPE_INSTANCE_GET_CLASS((obj), G_TYPE_RAW_INSTRUCTION, GRawInstructionClass)) - +#define G_TYPE_RAW_INSTRUCTION (g_raw_instruction_get_type()) -/* Définition générique d'une instruction brute d'architecture (instance) */ -typedef struct _GRawInstruction GRawInstruction; +DECLARE_GTYPE(GRawInstruction, g_raw_instruction, G, RAW_INSTRUCTION); -/* Définition générique d'une instruction brute d'architecture (classe) */ -typedef struct _GRawInstructionClass GRawInstructionClass; - - -/* Indique le type défini pour une instruction inconnue d'architecture. */ -GType g_raw_instruction_get_type(void);  /* Crée une instruction de type 'db/dw/etc' simple. */ -GArchInstruction *g_raw_instruction_new_from_value(const vmpa2t *, MemoryDataSize, uint64_t); +GArchInstruction *g_raw_instruction_new_from_value(GBinaryPortion *, const vmpa2t *, MemoryDataSize, uint64_t); -/* Crée une instruction de type 'db/dw/etc' pour un uleb128. */ -GArchInstruction *g_raw_instruction_new_uleb128(const GBinContent *content, vmpa2t *); +/*  Crée une instruction de type 'db/dw/etc' étendue. */ +GArchInstruction *g_raw_instruction_new_array(GBinaryPortion *, vmpa2t *, MemoryDataSize, const GBinContent *, size_t, SourceEndian); -/* Crée une instruction de type 'db/dw/etc' pour un sleb128. */ -GArchInstruction *g_raw_instruction_new_sleb128(const GBinContent *content, vmpa2t *); - -/* Crée une instruction de type 'db/dw/etc' étendue. */ -GArchInstruction *g_raw_instruction_new_array(const GBinContent *, MemoryDataSize, size_t, vmpa2t *, SourceEndian);  /* Drapeaux pour informations complémentaires */  typedef enum _RawInstrFlag @@ -74,6 +52,7 @@ typedef enum _RawInstrFlag  } RawInstrFlag; +  /* Marque l'instruction comme ne contenant que du bourrage. */  void g_raw_instruction_mark_as_padding(GRawInstruction *, bool); diff --git a/src/arch/instructions/undefined-int.h b/src/arch/instructions/undefined-int.h index a9b7627..faf0b4b 100644 --- a/src/arch/instructions/undefined-int.h +++ b/src/arch/instructions/undefined-int.h @@ -1,8 +1,8 @@  /* Chrysalide - Outil d'analyse de fichiers binaires - * undefined-int.h - prototypes pour la définition générique interne des instructions au comportement non défini + * undefined-int.h - prototypes pour la définition interne des instructions au comportement non défini   * - * Copyright (C) 2019 Cyrille Bagard + * Copyright (C) 2019-2025 Cyrille Bagard   *   *  This file is part of Chrysalide.   * @@ -27,42 +27,18 @@  #include "undefined.h"  #include "../instruction-int.h" -#include "../../glibext/objhole.h" -/* Informations glissées dans la structure GObject de GArchInstruction */ -typedef struct _undef_extra_data_t -{ -    /** -     * Le champ uid de la structure parente attendue conduit à une taille -     * alignée sur 2 octets, donc à une taille totale de 4 octets ce qui -     * représente la limite maximale de taille supportée. -     * -     * Pour 3 octets à la base, qui devraient laisser 8 - 1 octets disponbibles -     * en incluant le bit de verrouillage. -     * -     * On reproduit donc la structure instr_extra_data_t ici, en basculant -     * l'énumération InstrExpectedBehavior en champ de bits. -     */ - -    itid_t uid;                             /* Identifiant unique du type  */ -    ArchInstrFlag flags;                    /* Informations complémentaires*/ - -    unsigned int behavior : 2;              /* Conséquences réelles        */ - -} undef_extra_data_t; - -  /* Définition générique d'une instruction au comportement non défini (instance) */ -struct _GUndefInstruction +struct _GUndefinedInstruction  {      GArchInstruction parent;                /* A laisser en premier        */  };  /* Définition générique d'une instruction au comportement non défini (classe) */ -struct _GUndefInstructionClass +struct _GUndefinedInstructionClass  {      GArchInstructionClass parent;           /* A laisser en premier        */ @@ -73,15 +49,25 @@ struct _GUndefInstructionClass   * Accès aux informations éventuellement déportées.   */ -#if 1 //__SIZEOF_INT__ == __SIZEOF_LONG__ +/* Informations glissées dans la structure GObject de GArchInstruction */ +typedef struct _undef_extra_data_t +{ +    ARCH_INSTRUCTION_EXTRA_DATA + +    unsigned int behavior : 2;              /* Conséquences réelles        */ + +} undef_extra_data_t; + -#   define GET_UNDEF_INSTR_EXTRA(ins) ((undef_extra_data_t *)&((GArchInstruction *)ins)->extra) +#define GET_UNDEF_INSTR_EXTRA(op) \ +    GET_GOBJECT_EXTRA(op, undef_extra_data_t) -#else +#define SET_UNDEF_INSTR_EXTRA(op, data) \ +    SET_GOBJECT_EXTRA(op, undef_extra_data_t, data) -#   define GET_UNDEF_INSTR_EXTRA(ins) GET_GOBJECT_EXTRA(G_OBJECT(ins), undef_extra_data_t) -#endif +/* Met en place une instruction au comportement indéfini. */ +bool g_undefined_instruction_create(GUndefinedInstruction *, InstrExpectedBehavior); diff --git a/src/arch/instructions/undefined-ui.c b/src/arch/instructions/undefined-ui.c new file mode 100644 index 0000000..fbc0452 --- /dev/null +++ b/src/arch/instructions/undefined-ui.c @@ -0,0 +1,102 @@ + +/* Chrysalide - Outil d'analyse de fichiers binaires + * undefined-ui.c - opérandes représentant des instructions indéfinies sous forme graphique + * + * Copyright (C) 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 <http://www.gnu.org/licenses/>. + */ + + +#include "undefined-ui.h" + + +#include "../instruction.h" +#include "../../glibext/options/asm.h" + + + +/* Etablit dans une ligne de rendu le contenu représenté. */ +static void g_undefined_instruction_ui_populate_line(const GTokenGenerator *, size_t, size_t, GBufferLine *, void *); + + + +/****************************************************************************** +*                                                                             * +*  Paramètres  : iface = interface GLib à initialiser.                        * +*                                                                             * +*  Description : Procède à l'initialisation de l'interface de génération.     * +*                                                                             * +*  Retour      : -                                                            * +*                                                                             * +*  Remarques   : -                                                            * +*                                                                             * +******************************************************************************/ + +void g_undefined_instruction_ui_token_generator_iface_init(GTokenGeneratorInterface *iface) +{ +    iface->populate = g_undefined_instruction_ui_populate_line; + +} + + +/****************************************************************************** +*                                                                             * +*  Paramètres  : generator = générateur à utiliser pour l'impression.         * +*                index     = indice de cette même ligne dans le tampon global.* +*                repeat    = indice d'utilisations successives du générateur. * +*                line      = ligne de rendu à compléter.                      * +*                data      = éventuelle donnée complémentaire fournie.        * +*                                                                             * +*  Description : Etablit dans une ligne de rendu le contenu représenté.       * +*                                                                             * +*  Retour      : -                                                            * +*                                                                             * +*  Remarques   : -                                                            * +*                                                                             * +******************************************************************************/ + +static void g_undefined_instruction_ui_populate_line(const GTokenGenerator *generator, size_t index, size_t repeat, GBufferLine *line, void *data) +{ +    GArchInstruction *instr;                /* Version spécialisée         */ +    GBinContent *content;                   /* Contenu brut d'origine      */ +    mrange_t range;                         /* Emplacement couvert         */ +    char *key;                              /* Mot clef principal          */ + +    instr = G_ARCH_INSTRUCTION(generator); +    content = G_BIN_CONTENT(data); + +    /* Prologue */ + +    if (g_arch_instruction_get_range(instr, &range)) +    { +        g_buffer_line_fill_physical(line, ACO_PHYSICAL, MDS_32_BITS_UNSIGNED, get_mrange_addr(&range)); + +        g_buffer_line_fill_virtual(line, ACO_VIRTUAL, MDS_32_BITS_UNSIGNED, get_mrange_addr(&range)); + +        g_buffer_line_fill_content(line, ACO_BINARY, content, &range, VMPA_NO_PHYSICAL); + +    } + +    /* Instruction proprement dite */ + +    key = g_arch_instruction_get_keyword(instr); + +    g_buffer_line_append_text(line, ACO_ASSEMBLY_HEAD, TRT_ERROR, SL(key), NULL, G_OBJECT(instr)); + +    free(key); + +} diff --git a/src/arch/instructions/undefined-ui.h b/src/arch/instructions/undefined-ui.h new file mode 100644 index 0000000..8bc2d67 --- /dev/null +++ b/src/arch/instructions/undefined-ui.h @@ -0,0 +1,37 @@ + +/* Chrysalide - Outil d'analyse de fichiers binaires + * undefined-ui.h - prototypes pour les opérandes représentant des instructions indéfinies sous forme graphique + * + * Copyright (C) 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 <http://www.gnu.org/licenses/>. + */ + + +#ifndef _ARCH_OPERANDS_UNDEFINED_UI_H +#define _ARCH_OPERANDS_UNDEFINED_UI_H + + +#include "../../glibext/generator-int.h" + + + +/* Procède à l'initialisation de l'interface de génération. */ +void g_undefined_instruction_ui_token_generator_iface_init(GTokenGeneratorInterface *); + + + +#endif  /* _ARCH_OPERANDS_UNDEFINED_UI_H */ diff --git a/src/arch/instructions/undefined.c b/src/arch/instructions/undefined.c index 15c63e7..75df493 100644 --- a/src/arch/instructions/undefined.c +++ b/src/arch/instructions/undefined.c @@ -2,7 +2,7 @@  /* Chrysalide - Outil d'analyse de fichiers binaires   * undefined.c - instructions au comportement non défini   * - * Copyright (C) 2016-2019 Cyrille Bagard + * Copyright (C) 2016-2025 Cyrille Bagard   *   *  This file is part of Chrysalide.   * @@ -25,63 +25,42 @@  #include <assert.h> +#include <string.h>  #include <i18n.h>  #include "undefined-int.h" -#include "../../core/columns.h" +#include "../../glibext/serialize-int.h" +/* ------------------------- INSTRUCTION INCONNUE / DONNEES ------------------------- */ + +  /* Initialise la classe des instructions non définies. */ -static void g_undef_instruction_class_init(GUndefInstructionClass *); +static void g_undefined_instruction_class_init(GUndefinedInstructionClass *);  /* Initialise une instance d'instruction non définie. */ -static void g_undef_instruction_init(GUndefInstruction *); +static void g_undefined_instruction_init(GUndefinedInstruction *);  /* Supprime toutes les références externes. */ -static void g_undef_instruction_dispose(GUndefInstruction *); +static void g_undefined_instruction_dispose(GObject *);  /* Procède à la libération totale de la mémoire. */ -static void g_undef_instruction_finalize(GUndefInstruction *); - -/* Indique l'encodage d'une instruction de façon détaillée. */ -static const char *g_undef_instruction_get_encoding(const GUndefInstruction *); - -/* Fournit le nom humain de l'instruction manipulée. */ -static const char *g_undef_instruction_get_keyword(const GUndefInstruction *); - - - -/* -------------------- CONSERVATION SUR DISQUE DES INSTRUCTIONS -------------------- */ - - -/* Charge une instruction depuis une mémoire tampon. */ -static bool g_undef_instruction_unserialize(GUndefInstruction *, GAsmStorage *, GBinFormat *, packed_buffer_t *); - -/* Sauvegarde une instruction dans une mémoire tampon. */ -static bool g_undef_instruction_serialize(GUndefInstruction *, GAsmStorage *, packed_buffer_t *); - - - -/* ------------------------ OFFRE DE CAPACITES DE GENERATION ------------------------ */ - - -/* Ajoute à un tampon GLib le contenu de l'instance spécifiée. */ -static void g_undef_instruction_print(GUndefInstruction *, GBufferLine *, size_t, size_t, const GBinContent *); +static void g_undefined_instruction_finalize(GObject *);  /* --------------------- IMPLEMENTATION DES FONCTIONS DE CLASSE --------------------- */ -/* Charge un contenu depuis une mémoire tampon. */ -static bool g_undef_instruction_load(GUndefInstruction *, GObjectStorage *, packed_buffer_t *); +/* Indique l'encodage d'une instruction de façon détaillée. */ +static char *g_undefined_instruction_get_encoding(const GArchInstruction *); -/* Sauvegarde un contenu dans une mémoire tampon. */ -static bool g_undef_instruction_store(GUndefInstruction *, GObjectStorage *, packed_buffer_t *); +/* Fournit le nom humain de l'instruction manipulée. */ +static char *g_undefined_instruction_get_keyword(const GArchInstruction *); @@ -91,7 +70,9 @@ static bool g_undef_instruction_store(GUndefInstruction *, GObjectStorage *, pac  /* Indique le type défini pour une instruction au comportement non défini. */ -G_DEFINE_TYPE(GUndefInstruction, g_undef_instruction, G_TYPE_ARCH_INSTRUCTION); +G_DEFINE_TYPE_WITH_CODE(GUndefinedInstruction, g_undefined_instruction, G_TYPE_ARCH_INSTRUCTION, +                        G_IMPLEMENT_INTERFACE_IF_SYM(g_token_generator_get_type, g_undefined_instruction_ui_token_generator_iface_init)); +  /****************************************************************************** @@ -106,28 +87,20 @@ G_DEFINE_TYPE(GUndefInstruction, g_undef_instruction, G_TYPE_ARCH_INSTRUCTION);  *                                                                             *  ******************************************************************************/ -static void g_undef_instruction_class_init(GUndefInstructionClass *klass) +static void g_undefined_instruction_class_init(GUndefinedInstructionClass *klass)  {      GObjectClass *object;                   /* Autre version de la classe  */      GArchInstructionClass *instr;           /* Encore une autre vision...  */      object = G_OBJECT_CLASS(klass); -    object->dispose = (GObjectFinalizeFunc/* ! */)g_undef_instruction_dispose; -    object->finalize = (GObjectFinalizeFunc)g_undef_instruction_finalize; +    object->dispose = g_undefined_instruction_dispose; +    object->finalize = g_undefined_instruction_finalize;      instr = G_ARCH_INSTRUCTION_CLASS(klass); -    instr->get_encoding = (get_instruction_encoding_fc)g_undef_instruction_get_encoding; -    instr->get_keyword = (get_instruction_keyword_fc)g_undef_instruction_get_keyword; - -    instr->unserialize = (unserialize_instruction_fc)g_undef_instruction_unserialize; -    instr->serialize = (serialize_instruction_fc)g_undef_instruction_serialize; - -    instr->print = (print_instruction_fc)g_undef_instruction_print; - -    instr->load = (load_instruction_fc)g_undef_instruction_load; -    instr->store = (store_instruction_fc)g_undef_instruction_store; +    instr->get_encoding = g_undefined_instruction_get_encoding; +    instr->get_keyword = g_undefined_instruction_get_keyword;  } @@ -144,16 +117,22 @@ static void g_undef_instruction_class_init(GUndefInstructionClass *klass)  *                                                                             *  ******************************************************************************/ -static void g_undef_instruction_init(GUndefInstruction *instr) +static void g_undefined_instruction_init(GUndefinedInstruction *instr)  { -    GET_UNDEF_INSTR_EXTRA(instr)->behavior = IEB_UNDEFINED; +    undef_extra_data_t extra;               /* Données insérées à consulter*/ + +    extra = GET_UNDEF_INSTR_EXTRA(instr); + +    extra.behavior = IEB_UNDEFINED; + +    SET_UNDEF_INSTR_EXTRA(instr, &extra);  }  /******************************************************************************  *                                                                             * -*  Paramètres  : instr = instance d'objet GLib à traiter.                     * +*  Paramètres  : object = instance d'objet GLib à traiter.                    *  *                                                                             *  *  Description : Supprime toutes les références externes.                     *  *                                                                             * @@ -163,16 +142,16 @@ static void g_undef_instruction_init(GUndefInstruction *instr)  *                                                                             *  ******************************************************************************/ -static void g_undef_instruction_dispose(GUndefInstruction *instr) +static void g_undefined_instruction_dispose(GObject *object)  { -    G_OBJECT_CLASS(g_undef_instruction_parent_class)->dispose(G_OBJECT(instr)); +    G_OBJECT_CLASS(g_undefined_instruction_parent_class)->dispose(object);  }  /******************************************************************************  *                                                                             * -*  Paramètres  : instr = instance d'objet GLib à traiter.                     * +*  Paramètres  : object = instance d'objet GLib à traiter.                    *  *                                                                             *  *  Description : Procède à la libération totale de la mémoire.                *  *                                                                             * @@ -182,9 +161,9 @@ static void g_undef_instruction_dispose(GUndefInstruction *instr)  *                                                                             *  ******************************************************************************/ -static void g_undef_instruction_finalize(GUndefInstruction *instr) +static void g_undefined_instruction_finalize(GObject *object)  { -    G_OBJECT_CLASS(g_undef_instruction_parent_class)->finalize(G_OBJECT(instr)); +    G_OBJECT_CLASS(g_undefined_instruction_parent_class)->finalize(object);  } @@ -201,16 +180,14 @@ static void g_undef_instruction_finalize(GUndefInstruction *instr)  *                                                                             *  ******************************************************************************/ -GArchInstruction *g_undef_instruction_new(InstrExpectedBehavior behavior) +GArchInstruction *g_undefined_instruction_new(InstrExpectedBehavior behavior)  {      GArchInstruction *result;               /* Instruction à retourner     */ -    undef_extra_data_t *extra;              /* Données insérées à modifier */ -    result = g_object_new(G_TYPE_UNDEF_INSTRUCTION, NULL); +    result = g_object_new(G_TYPE_UNDEFINED_INSTRUCTION, NULL); -    extra = GET_UNDEF_INSTR_EXTRA(result); - -    extra->behavior = behavior; +    if (!g_undefined_instruction_create(G_UNDEFINED_INSTRUCTION(result), behavior)) +        g_clear_object(&result);      return result; @@ -219,90 +196,10 @@ GArchInstruction *g_undef_instruction_new(InstrExpectedBehavior behavior)  /******************************************************************************  *                                                                             * -*  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   : -                                                            * -*                                                                             * -******************************************************************************/ - -static const char *g_undef_instruction_get_encoding(const GUndefInstruction *instr) -{ -    const char *result;                     /* Description à retourner     */ - -    result = _("Undefined"); - -    return result; - -} - - -/****************************************************************************** +*  Paramètres  : instr    = instance à initialiser pleinement.                * +*                behavior = état réel du CPU après une passe de l'instruction.*  *                                                                             * -*  Paramètres  : instr = instruction d'assemblage à consulter.                * -*                                                                             * -*  Description : Fournit le nom humain de l'instruction manipulée.            * -*                                                                             * -*  Retour      : Mot clef de bas niveau.                                      * -*                                                                             * -*  Remarques   : -                                                            * -*                                                                             * -******************************************************************************/ - -const char *g_undef_instruction_get_keyword(const GUndefInstruction *instr) -{ -    const char *result;                     /* Désignation à retourner     */ -    undef_extra_data_t *extra;              /* Données insérées à consulter*/ - -    extra = GET_UNDEF_INSTR_EXTRA(instr); - -    switch (extra->behavior) -    { -        case IEB_NOP: -            result = "nop"; -            break; - -        case IEB_UNDEFINED: -            result = "undefined"; -            break; - -        case IEB_UNPREDICTABLE: -            result = "unpredictable"; -            break; - -        case IEB_RESERVED: -            result = "reserved"; -            break; - -        default: -            assert(false); -            result = NULL; -            break; - -    } - -    return result; - -} - - - -/* ---------------------------------------------------------------------------------- */ -/*                      CONSERVATION SUR DISQUE DES INSTRUCTIONS                      */ -/* ---------------------------------------------------------------------------------- */ - - -/****************************************************************************** -*                                                                             * -*  Paramètres  : instr   = instruction d'assemblage à consulter.              * -*                storage = mécanisme de sauvegarde à manipuler.               * -*                format  = format binaire chargé associé à l'architecture.    * -*                pbuf    = zone tampon à remplir.                             * -*                                                                             * -*  Description : Charge une instruction depuis une mémoire tampon.            * +*  Description : Met en place une instruction au comportement indéfini.       *  *                                                                             *  *  Retour      : Bilan de l'opération.                                        *  *                                                                             * @@ -310,122 +207,24 @@ const char *g_undef_instruction_get_keyword(const GUndefInstruction *instr)  *                                                                             *  ******************************************************************************/ -static bool g_undef_instruction_unserialize(GUndefInstruction *instr, GAsmStorage *storage, GBinFormat *format, packed_buffer_t *pbuf) +bool g_undefined_instruction_create(GUndefinedInstruction *instr, InstrExpectedBehavior behavior)  {      bool result;                            /* Bilan à retourner           */ -    GArchInstructionClass *parent;          /* Classe parente à consulter  */ -    undef_extra_data_t *extra;              /* Données insérées à consulter*/ -    uint8_t val;                            /* Champ de bits manipulé      */ - -    parent = G_ARCH_INSTRUCTION_CLASS(g_undef_instruction_parent_class); +    undef_extra_data_t extra;               /* Données insérées à modifier */ -    result = parent->unserialize(G_ARCH_INSTRUCTION(instr), storage, format, pbuf); +    result = true; -    if (result) -    { -        extra = GET_UNDEF_INSTR_EXTRA(instr); - -        LOCK_GOBJECT_EXTRA(extra); - -        result = extract_packed_buffer(pbuf, &val, sizeof(uint8_t), false); -        extra->behavior = val; - -        UNLOCK_GOBJECT_EXTRA(extra); - -    } - -    return result; - -} - - -/****************************************************************************** -*                                                                             * -*  Paramètres  : instr   = instruction d'assemblage à consulter.              * -*                storage = mécanisme de sauvegarde à manipuler.               * -*                pbuf    = zone tampon à remplir.                             * -*                                                                             * -*  Description : Sauvegarde une instruction dans une mémoire tampon.          * -*                                                                             * -*  Retour      : Bilan de l'opération.                                        * -*                                                                             * -*  Remarques   : -                                                            * -*                                                                             * -******************************************************************************/ - -static bool g_undef_instruction_serialize(GUndefInstruction *instr, GAsmStorage *storage, packed_buffer_t *pbuf) -{ -    bool result;                            /* Bilan à retourner           */ -    GArchInstructionClass *parent;          /* Classe parente à consulter  */ -    undef_extra_data_t *extra;              /* Données insérées à consulter*/ - -    parent = G_ARCH_INSTRUCTION_CLASS(g_undef_instruction_parent_class); - -    result = parent->serialize(G_ARCH_INSTRUCTION(instr), storage, pbuf); - -    if (result) -    { -        extra = GET_UNDEF_INSTR_EXTRA(instr); - -        LOCK_GOBJECT_EXTRA(extra); - -        result = extend_packed_buffer(pbuf, (uint8_t []){ extra->behavior }, sizeof(uint8_t), false); +    extra = GET_UNDEF_INSTR_EXTRA(instr); -        UNLOCK_GOBJECT_EXTRA(extra); +    extra.behavior = behavior; -    } +    SET_UNDEF_INSTR_EXTRA(instr, &extra);      return result;  } - -/* ---------------------------------------------------------------------------------- */ -/*                          OFFRE DE CAPACITES DE GENERATION                          */ -/* ---------------------------------------------------------------------------------- */ - - -/****************************************************************************** -*                                                                             * -*  Paramètres  : instr   = instruction d'assemblage à représenter.            * -*                line    = ligne de rendu à compléter.                        * -*                index   = indice de cette même ligne dans le tampon global.  * -*                repeat  = indice d'utilisations successives du générateur.   * -*                content = éventuel contenu binaire brut à imprimer.          * -*                                                                             * -*  Description : Ajoute à un tampon GLib le contenu de l'instance spécifiée.  * -*                                                                             * -*  Retour      : -                                                            * -*                                                                             * -*  Remarques   : -                                                            * -*                                                                             * -******************************************************************************/ - -static void g_undef_instruction_print(GUndefInstruction *instr, GBufferLine *line, size_t index, size_t repeat, const GBinContent *content) -{ -    GArchInstruction *base;                 /* Version de base             */ -    const char *key;                        /* Mot clef principal          */ -    size_t klen;                            /* Taille de ce mot clef       */ - -    base = G_ARCH_INSTRUCTION(instr); - -    g_buffer_line_fill_phys(line, DLC_PHYSICAL, MDS_32_BITS_UNSIGNED, get_mrange_addr(&base->range)); - -    g_buffer_line_fill_virt(line, DLC_VIRTUAL, MDS_32_BITS_UNSIGNED, get_mrange_addr(&base->range)); - -    g_buffer_line_fill_content(line, DLC_BINARY, content, &base->range, VMPA_NO_PHYSICAL); - -    /* Instruction proprement dite */ - -    key = g_arch_instruction_get_keyword(base); -    klen = strlen(key); - -    g_buffer_line_append_text(line, DLC_ASSEMBLY_HEAD, key, klen, RTT_ERROR, NULL); - -} - -  /******************************************************************************  *                                                                             *  *  Paramètres  : instr  = instruction à consulter.                            * @@ -438,18 +237,14 @@ static void g_undef_instruction_print(GUndefInstruction *instr, GBufferLine *lin  *                                                                             *  ******************************************************************************/ -InstrExpectedBehavior g_undef_instruction_get_behavior(const GUndefInstruction *instr) +InstrExpectedBehavior g_undefined_instruction_get_behavior(const GUndefinedInstruction *instr)  {      InstrExpectedBehavior result;           /* Comportement à retourner    */ -    undef_extra_data_t *extra;              /* Données insérées à consulter*/ +    undef_extra_data_t extra;               /* Données insérées à consulter*/      extra = GET_UNDEF_INSTR_EXTRA(instr); -    LOCK_GOBJECT_EXTRA(extra); - -    result = extra->behavior; - -    UNLOCK_GOBJECT_EXTRA(extra); +    result = extra.behavior;      return result; @@ -464,41 +259,21 @@ InstrExpectedBehavior g_undef_instruction_get_behavior(const GUndefInstruction *  /******************************************************************************  *                                                                             * -*  Paramètres  : instr   = élément GLib à constuire.                          * -*                storage = conservateur de données à manipuler ou NULL.       * -*                pbuf    = zone tampon à lire.                                * +*  Paramètres  : instr = instruction quelconque à consulter.                  *  *                                                                             * -*  Description : Charge un contenu depuis une mémoire tampon.                 * +*  Description : Indique l'encodage d'une instruction de façon détaillée.     *  *                                                                             * -*  Retour      : Bilan de l'opération.                                        * +*  Retour      : Description humaine de l'encodage utilisé.                   *  *                                                                             *  *  Remarques   : -                                                            *  *                                                                             *  ******************************************************************************/ -static bool g_undef_instruction_load(GUndefInstruction *instr, GObjectStorage *storage, packed_buffer_t *pbuf) +static char *g_undefined_instruction_get_encoding(const GArchInstruction *instr)  { -    bool result;                            /* Bilan à retourner           */ -    GArchInstructionClass *parent;          /* Classe parente à consulter  */ -    undef_extra_data_t *extra;              /* Données insérées à consulter*/ -    uint8_t val;                            /* Champ de bits manipulé      */ - -    parent = G_ARCH_INSTRUCTION_CLASS(g_undef_instruction_parent_class); - -    result = parent->load(G_ARCH_INSTRUCTION(instr), storage, pbuf); - -    if (result) -    { -        extra = GET_UNDEF_INSTR_EXTRA(instr); - -        LOCK_GOBJECT_EXTRA(extra); +    char *result;                           /* Description à retourner     */ -        result = extract_packed_buffer(pbuf, &val, sizeof(uint8_t), false); -        extra->behavior = val; - -        UNLOCK_GOBJECT_EXTRA(extra); - -    } +    result = strdup(_("Undefined"));      return result; @@ -507,37 +282,45 @@ static bool g_undef_instruction_load(GUndefInstruction *instr, GObjectStorage *s  /******************************************************************************  *                                                                             * -*  Paramètres  : instr   = élément GLib à consulter.                          * -*                storage = conservateur de données à manipuler ou NULL.       * -*                pbuf    = zone tampon à remplir.                             * +*  Paramètres  : instr = instruction d'assemblage à consulter.                *  *                                                                             * -*  Description : Sauvegarde un contenu dans une mémoire tampon.               * +*  Description : Fournit le nom humain de l'instruction manipulée.            *  *                                                                             * -*  Retour      : Bilan de l'opération.                                        * +*  Retour      : Mot clef de bas niveau.                                      *  *                                                                             *  *  Remarques   : -                                                            *  *                                                                             *  ******************************************************************************/ -static bool g_undef_instruction_store(GUndefInstruction *instr, GObjectStorage *storage, packed_buffer_t *pbuf) +static char *g_undefined_instruction_get_keyword(const GArchInstruction *instr)  { -    bool result;                            /* Bilan à retourner           */ -    GArchInstructionClass *parent;          /* Classe parente à consulter  */ -    undef_extra_data_t *extra;              /* Données insérées à consulter*/ - -    parent = G_ARCH_INSTRUCTION_CLASS(g_undef_instruction_parent_class); +    char *result;                           /* Désignation à retourner     */ +    undef_extra_data_t extra;               /* Données insérées à consulter*/ -    result = parent->store(G_ARCH_INSTRUCTION(instr), storage, pbuf); +    extra = GET_UNDEF_INSTR_EXTRA(instr); -    if (result) +    switch (extra.behavior)      { -        extra = GET_UNDEF_INSTR_EXTRA(instr); +        case IEB_NOP: +            result = strdup("nop"); +            break; -        LOCK_GOBJECT_EXTRA(extra); +        case IEB_UNDEFINED: +            result = strdup("undefined"); +            break; -        result = extend_packed_buffer(pbuf, (uint8_t []){ extra->behavior }, sizeof(uint8_t), false); +        case IEB_UNPREDICTABLE: +            result = strdup("unpredictable"); +            break; -        UNLOCK_GOBJECT_EXTRA(extra); +        case IEB_RESERVED: +            result = strdup("reserved"); +            break; + +        default: +            assert(false); +            result = NULL; +            break;      } diff --git a/src/arch/instructions/undefined.h b/src/arch/instructions/undefined.h index 8f35f35..d4b35f4 100644 --- a/src/arch/instructions/undefined.h +++ b/src/arch/instructions/undefined.h @@ -2,7 +2,7 @@  /* Chrysalide - Outil d'analyse de fichiers binaires   * undefined.h - prototypes pour les instructions au comportement non défini   * - * Copyright (C) 2016-2019 Cyrille Bagard + * Copyright (C) 2016-2025 Cyrille Bagard   *   *  This file is part of Chrysalide.   * @@ -25,27 +25,14 @@  #define _ARCH_INSTRUCTIONS_UNDEFINED_H -#include <glib-object.h> - -  #include "../instruction.h" -#include "../vmpa.h" - +#include "../../glibext/helpers.h" -#define G_TYPE_UNDEF_INSTRUCTION            g_undef_instruction_get_type() -#define G_UNDEF_INSTRUCTION(obj)            (G_TYPE_CHECK_INSTANCE_CAST((obj), G_TYPE_UNDEF_INSTRUCTION, GUndefInstruction)) -#define G_IS_UNDEF_INSTRUCTION(obj)         (G_TYPE_CHECK_INSTANCE_TYPE((obj), G_TYPE_UNDEF_INSTRUCTION)) -#define G_UNDEF_INSTRUCTION_CLASS(klass)    (G_TYPE_CHECK_CLASS_CAST((klass), G_TYPE_UNDEF_INSTRUCTION, GUndefInstructionClass)) -#define G_IS_UNDEF_INSTRUCTION_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE((klass), G_TYPE_UNDEF_INSTRUCTION)) -#define G_UNDEF_INSTRUCTION_GET_CLASS(obj)  (G_TYPE_INSTANCE_GET_CLASS((obj), G_TYPE_UNDEF_INSTRUCTION, GUndefInstructionClass)) +#define G_TYPE_UNDEFINED_INSTRUCTION (g_undefined_instruction_get_type()) -/* Définition générique d'une instruction au comportement non défini (instance) */ -typedef struct _GUndefInstruction GUndefInstruction; - -/* Définition générique d'une instruction au comportement non défini (classe) */ -typedef struct _GUndefInstructionClass GUndefInstructionClass; +DECLARE_GTYPE(GUndefinedInstruction, g_undefined_instruction, G, UNDEFINED_INSTRUCTION);  /* Etat précis de l'instruction */ @@ -59,14 +46,11 @@ typedef enum _InstrExpectedBehavior  } InstrExpectedBehavior; -/* Indique le type défini pour une instruction au comportement non défini. */ -GType g_undef_instruction_get_type(void); -  /* Crée une instruction au comportement nominalement indéfini. */ -GArchInstruction *g_undef_instruction_new(InstrExpectedBehavior); +GArchInstruction *g_undefined_instruction_new(InstrExpectedBehavior);  /* Indique le type de conséquences réél de l'instruction. */ -InstrExpectedBehavior g_undef_instruction_get_behavior(const GUndefInstruction *); +InstrExpectedBehavior g_undefined_instruction_get_behavior(const GUndefinedInstruction *); | 
