/* Chrysalide - Outil d'analyse de fichiers binaires
 * operand.c - gestion générique des opérandes
 *
 * Copyright (C) 2008-2020 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 "operand.h"


#include <assert.h>
#include <malloc.h>
#include <string.h>


#include "operand-int.h"
#include "storage.h"
#include "../analysis/storage/serialize-int.h"
#include "../common/fnv1a.h"
#include "../common/sort.h"
#include "../core/logs.h"
#include "../glibext/singleton-int.h"



/* ------------------------ DEFINITION D'OPERANDE QUELCONQUE ------------------------ */


/* Initialise la classe générique des opérandes. */
static void g_arch_operand_class_init(GArchOperandClass *);

/* Initialise une instance d'opérande d'architecture. */
static void g_arch_operand_init(GArchOperand *);

/* Procède à l'initialisation de l'interface de singleton. */
static void g_arch_operand_singleton_init(GSingletonCandidateInterface *);

/* Procède à l'initialisation de l'interface de sérialisation. */
static void g_arch_operand_serializable_init(GSerializableObjectInterface *);

/* Supprime toutes les références externes. */
static void g_arch_operand_dispose(GArchOperand *);

/* Procède à la libération totale de la mémoire. */
static void g_arch_operand_finalize(GArchOperand *);

/* Compare un opérande avec un autre. */
static int _g_arch_operand_compare(const GArchOperand *, const GArchOperand *, bool);



/* ------------------------ CONTROLE DU VOLUME DES INSTANCES ------------------------ */


/* Fournit une liste de candidats embarqués par un candidat. */
GArchOperand **g_arch_operand_list_inner_instances(const GArchOperand *, size_t *);

/* Met à jour une liste de candidats embarqués par un candidat. */
void g_arch_operand_update_inner_instances(GArchOperand *, GArchOperand **, size_t);

/* Fournit l'empreinte d'un candidat à une centralisation. */
static guint _g_arch_operand_hash(const GArchOperand *, bool);

/* Fournit l'empreinte d'un candidat à une centralisation. */
static guint g_arch_operand_hash(const GArchOperand *);

/* Détermine si deux candidats à l'unicité sont identiques. */
static gboolean g_arch_operand_is_equal(const GArchOperand *, const GArchOperand *);

/* Marque un candidat comme figé. */
static void g_arch_operand_set_read_only(GArchOperand *);

/* Indique si le candidat est figé. */
static bool g_arch_operand_is_read_only(GArchOperand *);



/* -------------------- CONSERVATION ET RECHARGEMENT DES DONNEES -------------------- */


/* Charge un contenu depuis une mémoire tampon. */
static bool _g_arch_operand_load(GArchOperand *, GObjectStorage *, packed_buffer_t *);

/* Charge un contenu depuis une mémoire tampon. */
static bool g_arch_operand_load(GArchOperand *, GObjectStorage *, packed_buffer_t *);

/* Sauvegarde un contenu dans une mémoire tampon. */
static bool _g_arch_operand_store(GArchOperand *, GObjectStorage *, packed_buffer_t *);

/* Sauvegarde un contenu dans une mémoire tampon. */
static bool g_arch_operand_store(GArchOperand *, GObjectStorage *, packed_buffer_t *);



/* ---------------------------------------------------------------------------------- */
/*                          DEFINITION D'OPERANDE QUELCONQUE                          */
/* ---------------------------------------------------------------------------------- */


/* Indique le type défini pour un opérande d'architecture. */
G_DEFINE_TYPE_WITH_CODE(GArchOperand, g_arch_operand, G_TYPE_OBJECT,
                        G_IMPLEMENT_INTERFACE(G_TYPE_SINGLETON_CANDIDATE, g_arch_operand_singleton_init)
                        G_IMPLEMENT_INTERFACE(G_TYPE_SERIALIZABLE_OBJECT, g_arch_operand_serializable_init));


/******************************************************************************
*                                                                             *
*  Paramètres  : klass = classe à initialiser.                                *
*                                                                             *
*  Description : Initialise la classe générique des opérandes.                *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static void g_arch_operand_class_init(GArchOperandClass *klass)
{
    GObjectClass *object;                   /* Autre version de la classe  */
    GArchOperandClass *operand;             /* Encore une autre vision...  */

    object = G_OBJECT_CLASS(klass);

    object->dispose = (GObjectFinalizeFunc/* ! */)g_arch_operand_dispose;
    object->finalize = (GObjectFinalizeFunc)g_arch_operand_finalize;

    operand = G_ARCH_OPERAND_CLASS(klass);

    operand->compare = (operand_compare_fc)_g_arch_operand_compare;

    operand->hash = _g_arch_operand_hash;

    operand->load = (load_operand_fc)_g_arch_operand_load;
    operand->store = (store_operand_fc)_g_arch_operand_store;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = instance à initialiser.                            *
*                                                                             *
*  Description : Initialise une instance d'opérande d'architecture.           *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static void g_arch_operand_init(GArchOperand *operand)
{
    operand_extra_data_t *extra;            /* Données insérées à modifier */

    extra = GET_ARCH_OP_EXTRA(operand);

    INIT_GOBJECT_EXTRA_LOCK(extra);

}


/******************************************************************************
*                                                                             *
*  Paramètres  : iface = interface GLib à initialiser.                        *
*                                                                             *
*  Description : Procède à l'initialisation de l'interface de singleton.      *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static void g_arch_operand_singleton_init(GSingletonCandidateInterface *iface)
{
    iface->list_inner = (list_inner_instances_fc)g_arch_operand_list_inner_instances;
    iface->update_inner = (update_inner_instances_fc)g_arch_operand_update_inner_instances;

    iface->hash = (hash_candidate_fc)g_arch_operand_hash;
    iface->is_equal = (is_candidate_equal_fc)g_arch_operand_is_equal;

    iface->set_ro = (set_candidate_ro_fc)g_arch_operand_set_read_only;
    iface->is_ro = (is_candidate_ro_fc)g_arch_operand_is_read_only;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : iface = interface GLib à initialiser.                        *
*                                                                             *
*  Description : Procède à l'initialisation de l'interface de sérialisation.  *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static void g_arch_operand_serializable_init(GSerializableObjectInterface *iface)
{
    iface->load = (load_serializable_object_cb)g_arch_operand_load;
    iface->store = (store_serializable_object_cb)g_arch_operand_store;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = instance d'objet GLib à traiter.                   *
*                                                                             *
*  Description : Supprime toutes les références externes.                     *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static void g_arch_operand_dispose(GArchOperand *operand)
{
    G_OBJECT_CLASS(g_arch_operand_parent_class)->dispose(G_OBJECT(operand));

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = instance d'objet GLib à traiter.                   *
*                                                                             *
*  Description : Procède à la libération totale de la mémoire.                *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static void g_arch_operand_finalize(GArchOperand *operand)
{
    G_OBJECT_CLASS(g_arch_operand_parent_class)->finalize(G_OBJECT(operand));

}


/******************************************************************************
*                                                                             *
*  Paramètres  : a    = premier opérande à consulter.                         *
*                b    = second opérande à consulter.                          *
*                lock = précise le besoin en verrouillage.                    *
*                                                                             *
*  Description : Compare un opérande avec un autre.                           *
*                                                                             *
*  Retour      : Bilan de la comparaison.                                     *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static int _g_arch_operand_compare(const GArchOperand *a, const GArchOperand *b, bool lock)
{
    int result;                             /* Bilan à faire remonter      */
    operand_extra_data_t *ea;               /* Données insérées à consulter*/
    operand_extra_data_t *eb;               /* Données insérées à consulter*/

    assert(!lock);

    ea = GET_ARCH_OP_EXTRA(a);
    eb = GET_ARCH_OP_EXTRA(b);

    result = sort_unsigned_long(ea->flags, eb->flags);

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : a = premier opérande à consulter.                            *
*                b = second opérande à consulter.                             *
*                                                                             *
*  Description : Compare un opérande avec un autre.                           *
*                                                                             *
*  Retour      : Bilan de la comparaison.                                     *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

int g_arch_operand_compare(const GArchOperand *a, const GArchOperand *b)
{
    int result;                             /* Bilan à faire remonter      */
    GType type_a;                           /* Type de l'object A          */
    GType type_b;                           /* Type de l'object B          */

    type_a = G_OBJECT_TYPE(G_OBJECT(a));
    type_b = G_OBJECT_TYPE(G_OBJECT(b));

    assert(sizeof(GType) <= sizeof(unsigned long));

    result = sort_unsigned_long(type_a, type_b);

    if (result == 0)
        result = G_ARCH_OPERAND_GET_CLASS(a)->compare(a, b, true);

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = opérande à consulter.                              *
*                target  = instruction à venir retrouver.                     *
*                                                                             *
*  Description : Détermine le chemin conduisant à un opérande interne.        *
*                                                                             *
*  Retour      : Chemin d'accès à l'opérande ou NULL en cas d'absence.        *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

char *g_arch_operand_find_inner_operand_path(const GArchOperand *operand, const GArchOperand *target)
{
    char *result;                           /* Chemin à retourner          */
    GArchOperandClass *class;               /* Classe associée à l'objet   */

    class = G_ARCH_OPERAND_GET_CLASS(operand);

    if (class->find_inner != NULL)
        result = class->find_inner(operand, target);

    else
        result = NULL;

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = opérande à consulter.                              *
*                path  = chemin d'accès à un opérande à retrouver.            *
*                                                                             *
*  Description : Obtient l'opérande correspondant à un chemin donné.          *
*                                                                             *
*  Retour      : Opérande trouvé ou NULL en cas d'échec.                      *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

GArchOperand *g_arch_operand_get_inner_operand_from_path(const GArchOperand *operand, const char *path)
{
    GArchOperand *result;                   /* Opérande trouvée à renvoyer */
    GArchOperandClass *class;               /* Classe associée à l'objet   */

    class = G_ARCH_OPERAND_GET_CLASS(operand);

    if (class->get_inner != NULL)
        result = class->get_inner(operand, path);

    else
        result = NULL;

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = opérande à traiter.                                *
*                line    = ligne tampon où imprimer l'opérande donné.         *
*                                                                             *
*  Description : Traduit un opérande en version humainement lisible.          *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

void g_arch_operand_print(const GArchOperand *operand, GBufferLine *line)
{
    G_ARCH_OPERAND_GET_CLASS(operand)->print(operand, line);

}


#ifdef INCLUDE_GTK_SUPPORT


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = opérande à consulter.                              *
*                binary  = informations relatives au binaire chargé.          *
*                                                                             *
*  Description : Construit un petit résumé concis de l'opérande.              *
*                                                                             *
*  Retour      : Chaîne de caractères à libérer après usage ou NULL.          *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

char *g_arch_operand_build_tooltip(const GArchOperand *operand, const GLoadedBinary *binary)
{
    char *result;                           /* Description à retourner     */
    GArchOperandClass *class;               /* Classe associée à l'objet   */

    class = G_ARCH_OPERAND_GET_CLASS(operand);

    if (class->build_tooltip != NULL)
        result = class->build_tooltip(operand, binary);
    else
        result = NULL;

    return result;

}


#endif


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = opérande à venir modifier.                         *
*                flag    = drapeau d'information complémentaire à planter.    *
*                lock    = indique un besoin de verrouillage des données.     *
*                                                                             *
*  Description : Ajoute une information complémentaire à un opérande.         *
*                                                                             *
*  Retour      : Bilan de l'opération.                                        *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

bool _g_arch_operand_set_flag(GArchOperand *operand, ArchOperandFlag flag, bool lock)
{
    bool result;                            /* Bilan à retourner           */
    operand_extra_data_t *extra;            /* Données insérées à modifier */

    assert(flag <= AOF_HIGH_USER);

    extra = GET_ARCH_OP_EXTRA(operand);

    if (lock)
        LOCK_GOBJECT_EXTRA(extra);

    result = !(extra->flags & flag);

    extra->flags |= flag;

    if (lock)
        UNLOCK_GOBJECT_EXTRA(extra);

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = opérande à venir modifier.                         *
*                flag    = drapeau d'information complémentaire à planter.    *
*                                                                             *
*  Description : Ajoute une information complémentaire à un opérande.         *
*                                                                             *
*  Retour      : Bilan de l'opération.                                        *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

bool g_arch_operand_set_flag(GArchOperand *operand, ArchOperandFlag flag)
{
    bool result;                            /* Bilan à retourner           */

    result = _g_arch_operand_set_flag(operand, flag, true);

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = opérande à venir modifier.                         *
*                flag    = drapeau d'information complémentaire à planter.    *
*                lock    = indique un besoin de verrouillage des données.     *
*                                                                             *
*  Description : Retire une information complémentaire à un opérande.         *
*                                                                             *
*  Retour      : Bilan de l'opération.                                        *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

bool _g_arch_operand_unset_flag(GArchOperand *operand, ArchOperandFlag flag, bool lock)
{
    bool result;                            /* Bilan à retourner           */
    operand_extra_data_t *extra;            /* Données insérées à modifier */

    assert(flag <= AOF_HIGH_USER);

    extra = GET_ARCH_OP_EXTRA(operand);

    LOCK_GOBJECT_EXTRA(extra);

    result = (extra->flags & flag);

    extra->flags &= ~flag;

    UNLOCK_GOBJECT_EXTRA(extra);

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = opérande à venir modifier.                         *
*                flag    = drapeau d'information complémentaire à planter.    *
*                                                                             *
*  Description : Retire une information complémentaire à un opérande.         *
*                                                                             *
*  Retour      : Bilan de l'opération.                                        *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

bool g_arch_operand_unset_flag(GArchOperand *operand, ArchOperandFlag flag)
{
    bool result;                            /* Bilan à retourner           */

    result = _g_arch_operand_unset_flag(operand, flag, true);

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = opérande à venir consulter.                        *
*                flag    = drapeau d'information à rechercher.                *
*                                                                             *
*  Description : Détermine si un opérande possède un fanion particulier.      *
*                                                                             *
*  Retour      : Bilan de la détection.                                       *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

bool g_arch_operand_has_flag(const GArchOperand *operand, ArchOperandFlag flag)
{
    bool result;                            /* Bilan à retourner           */
    operand_extra_data_t *extra;            /* Données insérées à modifier */

    assert(flag <= AOF_HIGH_USER);

    extra = GET_ARCH_OP_EXTRA(operand);

    LOCK_GOBJECT_EXTRA(extra);

    result = (extra->flags & flag);

    UNLOCK_GOBJECT_EXTRA(extra);

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = opérande à venir consulter.                        *
*                                                                             *
*  Description : Fournit les particularités de l'opérande.                    *
*                                                                             *
*  Retour      : Somme de tous les fanions associés à l'opérande.             *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

ArchOperandFlag g_arch_operand_get_flags(const GArchOperand *operand)
{
    ArchOperandFlag result;                 /* Fanions à retourner         */
    operand_extra_data_t *extra;            /* Données insérées à modifier */

    extra = GET_ARCH_OP_EXTRA(operand);

    LOCK_GOBJECT_EXTRA(extra);

    result = extra->flags;

    UNLOCK_GOBJECT_EXTRA(extra);

    return result;

}



/* ---------------------------------------------------------------------------------- */
/*                          CONTROLE DU VOLUME DES INSTANCES                          */
/* ---------------------------------------------------------------------------------- */


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = objet dont l'instance se veut unique.              *
*                count   = quantité d'instances à l'unicité internes.         *
*                                                                             *
*  Description : Fournit une liste de candidats embarqués par un candidat.    *
*                                                                             *
*  Retour      : Liste de candidats internes ou NULL si aucun.                *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

GArchOperand **g_arch_operand_list_inner_instances(const GArchOperand *operand, size_t *count)
{
    GArchOperand **result;                  /* Instances à retourner       */
    GArchOperandClass *class;               /* Classe associée à l'objet   */

    class = G_ARCH_OPERAND_GET_CLASS(operand);

    if (class->list_inner == NULL)
    {
        *count = 0;
        result = NULL;
    }

    else
        result = class->list_inner(operand, count);

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand   = objet dont l'instance se veut unique.            *
*                instances = liste de candidats internes devenus singletons.  *
*                count     = quantité d'instances à l'unicité internes.       *
*                                                                             *
*  Description : Met à jour une liste de candidats embarqués par un candidat. *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

void g_arch_operand_update_inner_instances(GArchOperand *operand, GArchOperand **instances, size_t count)
{
    GArchOperandClass *class;               /* Classe associée à l'objet   */

    class = G_ARCH_OPERAND_GET_CLASS(operand);

    if (class->update_inner == NULL)
        assert(class->list_inner == NULL);

    else
    {
        assert(class->list_inner != NULL);
        class->update_inner(operand, instances, count);
    }

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = objet dont l'instance se veut unique.              *
*                lock    = précise le besoin en verrouillage.                 *
*                                                                             *
*  Description : Fournit l'empreinte d'un candidat à une centralisation.      *
*                                                                             *
*  Retour      : Empreinte de l'élément représenté.                           *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static guint _g_arch_operand_hash(const GArchOperand *operand, bool lock)
{
    guint result;                           /* Valeur à retourner          */
    const char *name;                       /* Désignation du type d'object*/
    fnv64_t name_hash;                      /* Empreinte du nom            */
    operand_extra_data_t *extra;            /* Données insérées à modifier */

    assert(!lock);

    name = G_OBJECT_TYPE_NAME(G_OBJECT(operand));
    name_hash = fnv_64a_hash(name);

    result = (name_hash & 0xffffffff);
    result ^= (name_hash >> 32);

    extra = GET_ARCH_OP_EXTRA(operand);

    result ^= extra->flags;

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = objet dont l'instance se veut unique.              *
*                                                                             *
*  Description : Fournit l'empreinte d'un candidat à une centralisation.      *
*                                                                             *
*  Retour      : Empreinte de l'élément représenté.                           *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static guint g_arch_operand_hash(const GArchOperand *operand)
{
    guint result;                           /* Valeur à retourner          */
    GArchOperandClass *class;               /* Classe associée à l'objet   */

    class = G_ARCH_OPERAND_GET_CLASS(operand);

    result = class->hash(operand, true);

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = objet dont l'instance se veut unique.              *
*                other   = second élément à analyser.                         *
*                                                                             *
*  Description : Détermine si deux candidats à l'unicité sont identiques.     *
*                                                                             *
*  Retour      : Bilan de la comparaison.                                     *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static gboolean g_arch_operand_is_equal(const GArchOperand *operand, const GArchOperand *other)
{
    gboolean result;                        /* Bilan à renvoyer            */
    int ret;                                /* Bilan d'une comparaison     */

    ret = g_arch_operand_compare(operand, other);

    result = (ret == 0);

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = objet dont l'instance se veut unique.              *
*                                                                             *
*  Description : Marque un candidat comme figé.                               *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static void g_arch_operand_set_read_only(GArchOperand *operand)
{
    g_arch_operand_set_flag(operand, AOF_READ_ONLY);

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = objet dont l'instance se veut unique.              *
*                                                                             *
*  Description : Indique si le candidat est figé.                             *
*                                                                             *
*  Retour      : true si le contenu du candidat ne peut plus être modifié.    *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static bool g_arch_operand_is_read_only(GArchOperand *operand)
{
    bool result;                            /* Etat à retourner            */

    result = g_arch_operand_has_flag(operand, AOF_READ_ONLY);

    return result;

}



/* ---------------------------------------------------------------------------------- */
/*                      CONSERVATION ET RECHARGEMENT DES DONNEES                      */
/* ---------------------------------------------------------------------------------- */


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = élément GLib à constuire.                          *
*                storage = conservateur de données à manipuler ou NULL.       *
*                pbuf    = zone tampon à lire.                                *
*                                                                             *
*  Description : Charge un contenu depuis une mémoire tampon.                 *
*                                                                             *
*  Retour      : Bilan de l'opération.                                        *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static bool _g_arch_operand_load(GArchOperand *operand, GObjectStorage *storage, packed_buffer_t *pbuf)
{
    bool result;                            /* Bilan à retourner           */
    operand_extra_data_t *extra;            /* Données insérées à consulter*/
    uleb128_t value;                        /* Valeur ULEB128 à charger    */

    extra = GET_ARCH_OP_EXTRA(operand);

    LOCK_GOBJECT_EXTRA(extra);

    result = unpack_uleb128(&value, pbuf);

    if (result)
        extra->flags = value;

    UNLOCK_GOBJECT_EXTRA(extra);

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = élément GLib à constuire.                          *
*                storage = conservateur de données à manipuler ou NULL.       *
*                pbuf    = zone tampon à lire.                                *
*                                                                             *
*  Description : Charge un contenu depuis une mémoire tampon.                 *
*                                                                             *
*  Retour      : Bilan de l'opération.                                        *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static bool g_arch_operand_load(GArchOperand *operand, GObjectStorage *storage, packed_buffer_t *pbuf)
{
    bool result;                            /* Bilan à retourner           */
    GArchOperandClass *class;               /* Classe à activer            */

    class = G_ARCH_OPERAND_GET_CLASS(operand);

    result = class->load(operand, storage, pbuf);

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = élément GLib à consulter.                          *
*                storage = conservateur de données à manipuler ou NULL.       *
*                pbuf    = zone tampon à remplir.                             *
*                                                                             *
*  Description : Sauvegarde un contenu dans une mémoire tampon.               *
*                                                                             *
*  Retour      : Bilan de l'opération.                                        *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static bool _g_arch_operand_store(GArchOperand *operand, GObjectStorage *storage, packed_buffer_t *pbuf)
{
    bool result;                            /* Bilan à retourner           */
    operand_extra_data_t *extra;            /* Données insérées à consulter*/

    extra = GET_ARCH_OP_EXTRA(operand);

    LOCK_GOBJECT_EXTRA(extra);

    result = pack_uleb128((uleb128_t []){ extra->flags }, pbuf);

    UNLOCK_GOBJECT_EXTRA(extra);

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : operand = élément GLib à consulter.                          *
*                storage = conservateur de données à manipuler ou NULL.       *
*                pbuf    = zone tampon à remplir.                             *
*                                                                             *
*  Description : Sauvegarde un contenu dans une mémoire tampon.               *
*                                                                             *
*  Retour      : Bilan de l'opération.                                        *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static bool g_arch_operand_store(GArchOperand *operand, GObjectStorage *storage, packed_buffer_t *pbuf)
{
    bool result;                            /* Bilan à retourner           */
    GArchOperandClass *class;               /* Classe à activer            */

    class = G_ARCH_OPERAND_GET_CLASS(operand);

    result = class->store(operand, storage, pbuf);

    return result;

}