diff options
Diffstat (limited to 'src')
| -rw-r--r-- | src/glibext/Makefile.am | 6 | ||||
| -rw-r--r-- | src/glibext/comparable-int.h | 33 | ||||
| -rw-r--r-- | src/glibext/comparable.c | 139 | ||||
| -rw-r--r-- | src/glibext/comparable.h | 58 | ||||
| -rw-r--r-- | src/glibext/hashable-int.h | 47 | ||||
| -rw-r--r-- | src/glibext/hashable.c | 82 | ||||
| -rw-r--r-- | src/glibext/hashable.h | 42 | ||||
| -rw-r--r-- | src/glibext/singleton-int.h | 17 | ||||
| -rw-r--r-- | src/glibext/singleton.c | 179 | ||||
| -rw-r--r-- | src/glibext/singleton.h | 14 | ||||
| -rw-r--r-- | src/glibext/strbuilder-int.h | 2 | ||||
| -rw-r--r-- | src/glibext/strbuilder.c | 9 | ||||
| -rw-r--r-- | src/glibext/strbuilder.h | 2 | 
13 files changed, 342 insertions, 288 deletions
| diff --git a/src/glibext/Makefile.am b/src/glibext/Makefile.am index 738f5b3..2e9e86d 100644 --- a/src/glibext/Makefile.am +++ b/src/glibext/Makefile.am @@ -4,8 +4,6 @@ BUILT_SOURCES = chrysamarshal.h chrysamarshal.c resources.h resources.c  noinst_LTLIBRARIES  = libglibext.la libglibextui.la  # libglibext_la_SOURCES =						\ -# 	comparison-int.h						\ -# 	comparison.h comparison.c				\  # 	configuration-int.h						\  # 	configuration.h configuration.c			\  # 	gbinarycursor.h gbinarycursor.c			\ @@ -40,6 +38,10 @@ noinst_LTLIBRARIES  = libglibext.la libglibextui.la  libglibext_la_SOURCES =						\  	chrysamarshal.h chrysamarshal.c			\ +	comparable-int.h						\ +	comparable.h comparable.c				\ +	hashable-int.h							\ +	hashable.h hashable.c					\  	helpers.h								\  	objhole-int.h							\  	objhole.h objhole.c						\ diff --git a/src/glibext/comparable-int.h b/src/glibext/comparable-int.h index 446f25d..18972c2 100644 --- a/src/glibext/comparable-int.h +++ b/src/glibext/comparable-int.h @@ -1,8 +1,8 @@  /* Chrysalide - Outil d'analyse de fichiers binaires - * comparison-int.h - définitions internes propres aux opérations de comparaison d'objets + * comparable-int.h - définitions internes propres aux opérations de comparaison d'objets   * - * Copyright (C) 2022 Cyrille Bagard + * Copyright (C) 2022-2025 Cyrille Bagard   *   *  This file is part of Chrysalide.   * @@ -21,38 +21,27 @@   */ -#ifndef _GLIBEXT_COMPARISON_INT_H -#define _GLIBEXT_COMPARISON_INT_H +#ifndef _GLIBEXT_COMPARABLE_INT_H +#define _GLIBEXT_COMPARABLE_INT_H -#include "comparison.h" +#include "comparable.h" -/* Réalise une comparaison entre objets selon un critère précis. */ -typedef bool (* compare_rich_fc) (const GComparableItem *, const GComparableItem *, RichCmpOperation, bool *); +/* Réalise une comparaison étendue entre objets. */ +typedef int (* compare_object_fc) (const GComparableObject *, const GComparableObject *); -/* Instance d'élément comparable (interface) */ -struct _GComparableItemIface +/* Instance d'objet comparable (interface) */ +struct _GComparableObjectInterface  {      GTypeInterface base_iface;              /* A laisser en premier        */ -    compare_rich_fc cmp_rich;               /* Comparaison de façon précise*/ +    compare_object_fc compare;              /* Comparaison étendue         */  }; -/* Redéfinition */ -typedef GComparableItemIface GComparableItemInterface; - -/* Réalise une comparaison riche entre valeurs entière. */ -bool compare_rich_integer_values_signed(long long, long long, RichCmpOperation); - -/* Réalise une comparaison riche entre valeurs entière. */ -bool compare_rich_integer_values_unsigned(unsigned long long, unsigned long long, RichCmpOperation); - - - -#endif  /* _GLIBEXT_COMPARISON_INT_H */ +#endif  /* _GLIBEXT_COMPARABLE_INT_H */ diff --git a/src/glibext/comparable.c b/src/glibext/comparable.c index 8ce6941..95e7de7 100644 --- a/src/glibext/comparable.c +++ b/src/glibext/comparable.c @@ -1,8 +1,8 @@  /* Chrysalide - Outil d'analyse de fichiers binaires - * comparison.c - opérations de comparaison d'objets + * comparable.c - opérations de comparaison d'objets   * - * Copyright (C) 2022 Cyrille Bagard + * Copyright (C) 2022-2025 Cyrille Bagard   *   *  This file is part of Chrysalide.   * @@ -21,23 +21,20 @@   */ -#include "comparison.h" +#include "comparable.h" -#include <assert.h> - - -#include "comparison-int.h" +#include "comparable-int.h"  /* Procède à l'initialisation de l'interface de comparaison. */ -static void g_comparable_item_default_init(GComparableItemInterface *); +static void g_comparable_object_default_init(GComparableObjectInterface *);  /* Détermine le type d'une interface pour un objet comparable. */ -G_DEFINE_INTERFACE(GComparableItem, g_comparable_item, G_TYPE_OBJECT) +G_DEFINE_INTERFACE(GComparableObject, g_comparable_object, G_TYPE_OBJECT)  /****************************************************************************** @@ -52,35 +49,34 @@ G_DEFINE_INTERFACE(GComparableItem, g_comparable_item, G_TYPE_OBJECT)  *                                                                             *  ******************************************************************************/ -static void g_comparable_item_default_init(GComparableItemInterface *iface) +static void g_comparable_object_default_init(GComparableObjectInterface *iface)  { +    iface->compare = NULL;  }  /******************************************************************************  *                                                                             * -*  Paramètres  : item   = premier objet à consulter pour une comparaison.     * +*  Paramètres  : object = premier objet à consulter pour une comparaison.     *  *                other  = second objet à consulter pour une comparaison.      * -*                op     = opération de comparaison à réaliser.                * -*                status = bilan des opérations de comparaison. [OUT]          *  *                                                                             * -*  Description : Réalise une comparaison entre objets selon un critère précis.* +*  Description : Réalise une comparaison étendue entre objets.                *  *                                                                             * -*  Retour      : true si la comparaison a pu être effectuée, false sinon.     * +*  Retour      : Bilan de la comparaison.                                     *  *                                                                             *  *  Remarques   : -                                                            *  *                                                                             *  ******************************************************************************/ -bool g_comparable_item_compare_rich(const GComparableItem *item, const GComparableItem *other, RichCmpOperation op, bool *status) +int g_comparable_object_compare(const GComparableObject *object, const GComparableObject *other)  { -    bool result;                            /* Etat à retourner            */ -    GComparableItemIface *iface;            /* Interface utilisée          */ +    int result;                             /* Bilan à retourner           */ +    GComparableObjectInterface *iface;      /* Interface utilisée          */ -    iface = G_COMPARABLE_ITEM_GET_IFACE(item); +    iface = G_COMPARABLE_OBJECT_GET_IFACE(object); -    result = iface->cmp_rich(item, other, op, status); +    result = iface->compare(object, other);      return result; @@ -89,110 +85,25 @@ bool g_comparable_item_compare_rich(const GComparableItem *item, const GComparab  /******************************************************************************  *                                                                             * -*  Paramètres  : a  = premier élément à consulter pour une comparaison.       * -*                b  = second objet à consulter pour une comparaison.          * -*                op = opération de comparaison à réaliser.                    * -*                                                                             * -*  Description : Réalise une comparaison riche entre valeurs entière.         * -*                                                                             * -*  Retour      : Bilan des opérations de comparaison.                         * -*                                                                             * -*  Remarques   : -                                                            * -*                                                                             * -******************************************************************************/ - -bool compare_rich_integer_values_signed(long long a, long long b, RichCmpOperation op) -{ -    bool result;                            /* Bilan  à retourner          */ - -    switch (op) -    { -        case RCO_LT: -            result = (a < b); -            break; - -        case RCO_LE: -            result = (a <= b); -            break; - -        case RCO_EQ: -            result = (a == b); -            break; - -        case RCO_NE: -            result = (a != b); -            break; - -        case RCO_GT: -            result = (a > b); -            break; - -        case RCO_GE: -            result = (a >= b); -            break; - -        default: -            assert(false); -            result = false; -            break; - -    } - -    return result; - -} - - -/****************************************************************************** -*                                                                             * -*  Paramètres  : a  = premier élément à consulter pour une comparaison.       * -*                b  = second objet à consulter pour une comparaison.          * -*                op = opération de comparaison à réaliser.                    * +*  Paramètres  : object = premier objet à consulter pour une comparaison.     * +*                other  = second objet à consulter pour une comparaison.      *  *                                                                             * -*  Description : Réalise une comparaison riche entre valeurs entière.         * +*  Description : Détermine si deux objets sont fonctionnellement identiques.  *  *                                                                             * -*  Retour      : Bilan des opérations de comparaison.                         * +*  Retour      : Bilan de la comparaison.                                     *  *                                                                             *  *  Remarques   : -                                                            *  *                                                                             *  ******************************************************************************/ -bool compare_rich_integer_values_unsigned(unsigned long long a, unsigned long long b, RichCmpOperation op) +gboolean g_comparable_object_is_equal(const GComparableObject *object, const GComparableObject *other)  { -    bool result;                            /* Bilan  à retourner          */ - -    switch (op) -    { -        case RCO_LT: -            result = (a < b); -            break; - -        case RCO_LE: -            result = (a <= b); -            break; - -        case RCO_EQ: -            result = (a == b); -            break; - -        case RCO_NE: -            result = (a != b); -            break; - -        case RCO_GT: -            result = (a > b); -            break; - -        case RCO_GE: -            result = (a >= b); -            break; +    gboolean result;                        /* Bilan à renvoyer            */ +    int ret;                                /* Bilan d'une comparaison     */ -        default: -            assert(false); -            result = false; -            break; +    ret = g_comparable_object_compare(object, other); -    } +    result = (ret == 0);      return result; diff --git a/src/glibext/comparable.h b/src/glibext/comparable.h index 8d43210..7a652ff 100644 --- a/src/glibext/comparable.h +++ b/src/glibext/comparable.h @@ -1,8 +1,8 @@  /* Chrysalide - Outil d'analyse de fichiers binaires - * comparison.h - prototypes pour les opérations de comparaison d'objets + * comparable.h - prototypes pour les opérations de comparaison d'objets   * - * Copyright (C) 2022 Cyrille Bagard + * Copyright (C) 2022-2025 Cyrille Bagard   *   *  This file is part of Chrysalide.   * @@ -21,60 +21,26 @@   */ -#ifndef _GLIBEXT_COMPARISON_H -#define _GLIBEXT_COMPARISON_H +#ifndef _GLIBEXT_COMPARABLE_H +#define _GLIBEXT_COMPARABLE_H -#include <glib-object.h> -#include <stdbool.h> +#include "helpers.h" -#define G_TYPE_COMPARABLE_ITEM             (g_comparable_item_get_type()) -#define G_COMPARABLE_ITEM(obj)             (G_TYPE_CHECK_INSTANCE_CAST((obj), G_TYPE_COMPARABLE_ITEM, GComparableItem)) -#define G_COMPARABLE_ITEM_CLASS(vtable)    (G_TYPE_CHECK_CLASS_CAST((vtable), G_TYPE_COMPARABLE_ITEM, GComparableItemIface)) -#define G_IS_COMPARABLE_ITEM(obj)          (G_TYPE_CHECK_INSTANCE_TYPE((obj), G_TYPE_COMPARABLE_ITEM)) -#define G_IS_COMPARABLE_ITEM_CLASS(vtable) (G_TYPE_CHECK_CLASS_TYPE((vtable), G_TYPE_COMPARABLE_ITEM)) -#define G_COMPARABLE_ITEM_GET_IFACE(inst)  (G_TYPE_INSTANCE_GET_INTERFACE((inst), G_TYPE_COMPARABLE_ITEM, GComparableItemIface)) +#define G_TYPE_COMPARABLE_OBJECT (g_comparable_object_get_type()) +DECLARE_INTERFACE(GComparableObject, g_comparable_object, G, COMPARABLE_OBJECT); -/* Instance d'élément comparable (coquille vide) */ -typedef struct _GComparableItem GComparableItem; -/* Instance d'élément comparable (interface) */ -typedef struct _GComparableItemIface GComparableItemIface; +/* Réalise une comparaison étendue entre objets. */ +int g_comparable_object_compare(const GComparableObject *, const GComparableObject *); -/* Modes de comparaison */ -typedef enum _RichCmpOperation -{ -    RCO_LT,                                 /* Equivalent de '<'           */ -    RCO_LE,                                 /* Equivalent de '<='          */ -    RCO_EQ,                                 /* Equivalent de '=='          */ -    RCO_NE,                                 /* Equivalent de '!='          */ -    RCO_GT,                                 /* Equivalent de '>'           */ -    RCO_GE,                                 /* Equivalent de '>°'          */ +/* Détermine si deux objets sont fonctionnellement identiques. */ +gboolean g_comparable_object_is_equal(const GComparableObject *, const GComparableObject *); -} RichCmpOperation; -/* Détermination d'un besoin de comparaison supplémentaire */ -#define STATUS_NOT_EQUAL(_s, _o)                            \ -    ({                                                      \ -        bool __result;                                      \ -        if (_o == RCO_LE || _o == RCO_EQ || _o == RCO_GE)   \ -            __result = !_s;                                 \ -        else                                                \ -            __result = _s;                                  \ -        __result;                                           \ -    }) - -/* Détermine le type d'une interface pour un objet comparable. */ -GType g_comparable_item_get_type(void) G_GNUC_CONST; - -/* Réalise une comparaison entre objets selon un critère précis. */ -bool g_comparable_item_compare_rich(const GComparableItem *, const GComparableItem *, RichCmpOperation, bool *); - - - -#endif  /* _GLIBEXT_COMPARISON_H */ +#endif  /* _GLIBEXT_COMPARABLE_H */ diff --git a/src/glibext/hashable-int.h b/src/glibext/hashable-int.h new file mode 100644 index 0000000..f8a85e1 --- /dev/null +++ b/src/glibext/hashable-int.h @@ -0,0 +1,47 @@ + +/* Chrysalide - Outil d'analyse de fichiers binaires + * hashable-int.h - définitions internes propres aux calculs de l'empreinte d'un objet + * + * 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 _GLIBEXT_HASHABLE_INT_H +#define _GLIBEXT_HASHABLE_INT_H + + +#include "hashable.h" + + + +/* Calcule l'empreinte sur 32 bits d'un objet. */ +typedef guint (* hash_object_fc) (const GHashableObject *); + + +/* Instance d'objet visant à être unique (interface) */ +struct _GHashableObjectInterface +{ +    GTypeInterface base_iface;              /* A laisser en premier        */ + +    hash_object_fc hash;                    /* Réduction en valeur         */ + +}; + + + +#endif  /* _GLIBEXT_HASHABLE_INT_H */ diff --git a/src/glibext/hashable.c b/src/glibext/hashable.c new file mode 100644 index 0000000..f988a90 --- /dev/null +++ b/src/glibext/hashable.c @@ -0,0 +1,82 @@ + +/* Chrysalide - Outil d'analyse de fichiers binaires + * hashable.c - calculs de l'empreinte d'un objet + * + * 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 Foobar.  If not, see <http://www.gnu.org/licenses/>. + */ + + +#include "hashable.h" + + +#include "hashable-int.h" + + + +/* Procède à l'initialisation de l'interface de détermination. */ +static void g_hashable_object_default_init(GHashableObjectInterface *); + + + +/* Détermine le type d'une interface pour la réduction d'un objet à une valeur. */ +G_DEFINE_INTERFACE(GHashableObject, g_hashable_object, G_TYPE_OBJECT) + + +/****************************************************************************** +*                                                                             * +*  Paramètres  : iface = interface GLib à initialiser.                        * +*                                                                             * +*  Description : Procède à l'initialisation de l'interface de détermination.  * +*                                                                             * +*  Retour      : -                                                            * +*                                                                             * +*  Remarques   : -                                                            * +*                                                                             * +******************************************************************************/ + +static void g_hashable_object_default_init(GHashableObjectInterface *iface) +{ +    iface->hash = NULL; + +} + + +/****************************************************************************** +*                                                                             * +*  Paramètres  : object = objet dont l'instance est à consulter.              * +*                                                                             * +*  Description : Calcule l'empreinte sur 32 bits d'un objet.                  * +*                                                                             * +*  Retour      : Valeur de représentation, unique pour l'objet ou non.        * +*                                                                             * +*  Remarques   : -                                                            * +*                                                                             * +******************************************************************************/ + +guint g_hashable_object_hash(const GHashableObject *object) +{ +    guint result;                       /* Valeur à retourner              */ +    GHashableObjectInterface *iface;    /* Interface utilisée              */ + +    iface = G_HASHABLE_OBJECT_GET_IFACE(object); + +    result = iface->hash(object); + +    return result; + +} diff --git a/src/glibext/hashable.h b/src/glibext/hashable.h new file mode 100644 index 0000000..165c744 --- /dev/null +++ b/src/glibext/hashable.h @@ -0,0 +1,42 @@ + +/* Chrysalide - Outil d'analyse de fichiers binaires + * hashable.h - prototypes pour les calculs de l'empreinte d'un objet + * + * 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 Foobar.  If not, see <http://www.gnu.org/licenses/>. + */ + + +#ifndef _GLIBEXT_HASHABLE_H +#define _GLIBEXT_HASHABLE_H + + +#include "helpers.h" + + + +#define G_TYPE_HASHABLE_OBJECT (g_hashable_object_get_type()) + +DECLARE_INTERFACE(GHashableObject, g_hashable_object, G, HASHABLE_OBJECT); + + +/* Calcule l'empreinte sur 32 bits d'un objet. */ +guint g_hashable_object_hash(const GHashableObject *); + + + +#endif  /* _GLIBEXT_HASHABLE_H */ diff --git a/src/glibext/singleton-int.h b/src/glibext/singleton-int.h index 9212e2e..747e64a 100644 --- a/src/glibext/singleton-int.h +++ b/src/glibext/singleton-int.h @@ -38,11 +38,14 @@ typedef GSingletonCandidate ** (* list_inner_instances_fc) (const GSingletonCand  /* Met à jour une liste de candidats embarqués par un candidat. */  typedef void (* update_inner_instances_fc) (GSingletonCandidate *, GSingletonCandidate **, size_t); -/* Fournit l'empreinte d'un candidat à une centralisation. */ -typedef guint (* hash_candidate_fc) (const GSingletonCandidate *); +/* Marque un candidat comme figé. */ +typedef void (* mark_candidate_as_ro_fc) (GSingletonCandidate *); -/* Détermine si deux candidats à l'unicité sont identiques. */ -typedef gboolean (* is_candidate_equal_fc) (const GSingletonCandidate *, const GSingletonCandidate *); +/* Indique si le candidat est figé. */ +typedef bool (* is_candidate_ro_fc) (const GSingletonCandidate *); + +/* Crée une copie modifiable d'un object unique. */ +typedef GSingletonCandidate * (* dup_candidate_fc) (const GSingletonCandidate *);  /* Instance d'objet visant à être unique (interface) */ @@ -53,8 +56,10 @@ struct _GSingletonCandidateInterface      list_inner_instances_fc list_inner;     /* Récupération d'internes     */      update_inner_instances_fc update_inner; /* Mise à jour des éléments    */ -    hash_candidate_fc hash;                 /* Prise d'empreinte           */ -    is_candidate_equal_fc is_equal;         /* Comparaison                 */ +    mark_candidate_as_ro_fc mark_as_ro;     /* Bascule en mode figé        */ +    is_candidate_ro_fc is_ro;               /* Consultation de l'état      */ + +    dup_candidate_fc dup;                   /* Création de copie modifiable*/  }; diff --git a/src/glibext/singleton.c b/src/glibext/singleton.c index 65b83e7..4930323 100644 --- a/src/glibext/singleton.c +++ b/src/glibext/singleton.c @@ -26,9 +26,10 @@  #include <assert.h>  #include <malloc.h> -#include <stdbool.h> +#include "comparable.h" +#include "hashable.h"  #include "singleton-int.h" @@ -39,11 +40,14 @@  /* Procède à l'initialisation de l'interface de rassemblement. */  static void g_singleton_candidate_default_init(GSingletonCandidateInterface *); +/* Fournit une liste de candidats embarqués par un candidat. */ +static GSingletonCandidate **g_singleton_candidate_list_inner_instances(const GSingletonCandidate *, size_t *); +  /* Met à jour une liste de candidats embarqués par un candidat. */  static void g_singleton_candidate_update_inner_instances(GSingletonCandidate *, GSingletonCandidate **, size_t); -/* Détermine si deux candidats à l'unicité sont identiques. */ -static gboolean _g_singleton_candidate_is_equal(GSingletonCandidate *, GSingletonCandidate *, GList **); +/* Marque un candidat comme figé. */ +static void g_singleton_candidate_mark_as_read_only(GSingletonCandidate *); @@ -70,7 +74,9 @@ static void g_singleton_factory_finalize(GSingletonFactory *);  /* Détermine le type d'une interface pour la constitution d'objets uniques. */ -G_DEFINE_INTERFACE(GSingletonCandidate, g_singleton_candidate, G_TYPE_OBJECT) +G_DEFINE_INTERFACE_WITH_CODE(GSingletonCandidate, g_singleton_candidate, G_TYPE_OBJECT, +                             g_type_interface_add_prerequisite(g_define_type_id, G_TYPE_HASHABLE_OBJECT); +                             g_type_interface_add_prerequisite(g_define_type_id, G_TYPE_COMPARABLE_OBJECT))  /****************************************************************************** @@ -87,6 +93,13 @@ G_DEFINE_INTERFACE(GSingletonCandidate, g_singleton_candidate, G_TYPE_OBJECT)  static void g_singleton_candidate_default_init(GSingletonCandidateInterface *iface)  { +    iface->list_inner = NULL; +    iface->update_inner = NULL; + +    iface->mark_as_ro = NULL; +    iface->is_ro = NULL; + +    iface->dup = NULL;  } @@ -113,8 +126,11 @@ GSingletonCandidate **g_singleton_candidate_list_inner_instances(const GSingleto      if (iface->list_inner == NULL)      { +        assert(iface->update_inner == NULL); +          *count = 0;          result = NULL; +      }      else @@ -161,17 +177,16 @@ static void g_singleton_candidate_update_inner_instances(GSingletonCandidate *ca  *                                                                             *  *  Paramètres  : candidate = objet dont l'instance se veut unique.            *  *                                                                             * -*  Description : Fournit l'empreinte d'un candidat à une centralisation.      * +*  Description : Marque un candidat comme figé.                               *  *                                                                             * -*  Retour      : Empreinte de l'élément représenté.                           * +*  Retour      : -                                                            *  *                                                                             *  *  Remarques   : -                                                            *  *                                                                             *  ******************************************************************************/ -guint g_singleton_candidate_hash(GSingletonCandidate *candidate) +static void g_singleton_candidate_mark_as_read_only(GSingletonCandidate *candidate)  { -    guint result;                           /* Valeur à retourner          */      GSingletonCandidateInterface *iface;    /* Interface utilisée          */      GSingletonCandidate **children;         /* Instances internes          */      size_t count;                           /* Quantité de ces instances   */ @@ -179,103 +194,62 @@ guint g_singleton_candidate_hash(GSingletonCandidate *candidate)      iface = G_SINGLETON_CANDIDATE_GET_IFACE(candidate); -    result = iface->hash(candidate); +    iface->mark_as_ro(candidate);      children = g_singleton_candidate_list_inner_instances(candidate, &count);      for (i = 0; i < count; i++)      { -        result ^= g_singleton_candidate_hash(children[i]); -        unref_object(children[i]); +        g_singleton_candidate_mark_as_read_only(children[i]); +        unref_object(G_OBJECT(children[i]));      }      if (children != NULL)          free(children); -    return result; -  }  /******************************************************************************  *                                                                             *  *  Paramètres  : candidate = objet dont l'instance se veut unique.            * -*                other     = second élément à analyser.                       * -*                processed = liste de candidats déjà traités.                 *  *                                                                             * -*  Description : Détermine si deux candidats à l'unicité sont identiques.     * +*  Description : Indique si le candidat est figé.                             *  *                                                                             * -*  Retour      : Bilan de la comparaison.                                     * +*  Retour      : true si le contenu du candidat ne peut plus être modifié.    *  *                                                                             *  *  Remarques   : -                                                            *  *                                                                             *  ******************************************************************************/ -static gboolean _g_singleton_candidate_is_equal(GSingletonCandidate *candidate, GSingletonCandidate *other, GList **processed) +bool g_singleton_candidate_is_read_only(const GSingletonCandidate *candidate)  { -    gboolean result;                        /* Bilan à renvoyer            */ -    GList *skip;                            /* Détection de boucle         */ +    bool result;                            /* Etat à retourner            */      GSingletonCandidateInterface *iface;    /* Interface utilisée          */ -    GSingletonCandidate **children[2];      /* Instances internes          */ -    size_t count[2];                        /* Quantité de ces instances   */ +#ifndef NDEBUG +    GSingletonCandidate **children;         /* Instances internes          */ +    size_t count;                           /* Quantité de ces instances   */      size_t i;                               /* Boucle de parcours          */ +#endif -    skip = g_list_find(processed[0], candidate); - -    if (skip != NULL) -        result = (g_list_find(processed[1], other) != NULL); - -    else -    { -        iface = G_SINGLETON_CANDIDATE_GET_IFACE(candidate); - -        result = iface->is_equal(candidate, other); - -        processed[0] = g_list_append(processed[0], candidate); -        processed[1] = g_list_append(processed[1], other); - -        if (!result) -            goto done; - -        children[0] = g_singleton_candidate_list_inner_instances(candidate, &count[0]); -        children[1] = g_singleton_candidate_list_inner_instances(other, &count[1]); - -        if (count[0] != count[1]) -        { -            for (i = 0; i < count[0]; i++) -                g_object_unref(G_OBJECT(children[0][i])); - -            for (i = 0; i < count[1]; i++) -                g_object_unref(G_OBJECT(children[1][i])); - -        } - -        else -        { -            for (i = 0; i < count[0] && result; i++) -            { -                result = _g_singleton_candidate_is_equal(children[0][i], children[1][i], processed); -                g_object_unref(G_OBJECT(children[0][i])); -                g_object_unref(G_OBJECT(children[1][i])); -            } - -            for (; i < count[0]; i++) -            { -                g_object_unref(G_OBJECT(children[0][i])); -                g_object_unref(G_OBJECT(children[1][i])); -            } +    iface = G_SINGLETON_CANDIDATE_GET_IFACE(candidate); -            if (children[0] != NULL) -                free(children[0]); +    result = iface->is_ro(candidate); -            if (children[1] != NULL) -                free(children[1]); +#ifndef NDEBUG -        } +    children = g_singleton_candidate_list_inner_instances(candidate, &count); +    for (i = 0; i < count; i++) +    { +        assert(result == g_singleton_candidate_is_read_only(children[i])); +        unref_object(G_OBJECT(children[i]));      } - done: +    if (children != NULL) +        free(children); + +#endif      return result; @@ -285,31 +259,60 @@ static gboolean _g_singleton_candidate_is_equal(GSingletonCandidate *candidate,  /******************************************************************************  *                                                                             *  *  Paramètres  : candidate = objet dont l'instance se veut unique.            * -*                other     = second élément à analyser.                       *  *                                                                             * -*  Description : Détermine si deux candidats à l'unicité sont identiques.     * +*  Description : Crée une copie modifiable d'un object unique.                *  *                                                                             * -*  Retour      : Bilan de la comparaison.                                     * +*  Retour      : Nouvelle instance mise en place.                             *  *                                                                             *  *  Remarques   : -                                                            *  *                                                                             *  ******************************************************************************/ -gboolean g_singleton_candidate_is_equal(GSingletonCandidate *candidate, GSingletonCandidate *other) +GSingletonCandidate *g_singleton_candidate_dup(const GSingletonCandidate *candidate)  { -    gboolean result;                        /* Bilan à renvoyer            */ -    GList *processed[2];                    /* Suivi des traitements       */ +    GSingletonCandidate *result;            /* Instance à retourner        */ +    GSingletonCandidateInterface *iface;    /* Interface utilisée          */ +    size_t count;                           /* Quantité d'objets internes  */ +    GSingletonCandidate **children;         /* Liste d'instances internes  */ +    size_t i;                               /* Boucle de parcours          */ +    GSingletonCandidate **new_children;     /* Nouvelle liste d'instances  */ + +    iface = G_SINGLETON_CANDIDATE_GET_IFACE(candidate); + +    result = iface->dup(candidate); + +    assert(!g_singleton_candidate_is_read_only(result)); + +    children = g_singleton_candidate_list_inner_instances(candidate, &count); + +    if (count > 0) +    { +        new_children = malloc(count * sizeof(GSingletonCandidate *)); + +        for (i = 0; i < count; i++) +        { +            new_children[i] = g_singleton_candidate_dup(children[i]); +             +            assert(!g_singleton_candidate_is_read_only(new_children[i])); + +        } + +        g_singleton_candidate_update_inner_instances(result, new_children, count); -    processed[0] = NULL; -    processed[1] = NULL; +        for (i = 0; i < count; i++) +        { +            unref_object(G_OBJECT(new_children[i])); +            unref_object(G_OBJECT(children[i])); +        } + +        free(new_children); -    result = _g_singleton_candidate_is_equal(candidate, other, processed); +    } -    assert(processed[0] != NULL); -    assert(processed[1] != NULL); +    if (children != NULL) +        free(children); -    g_list_free(processed[0]); -    g_list_free(processed[1]); +    assert(G_OBJECT_TYPE(result) == G_OBJECT_TYPE(candidate));      return result; @@ -364,8 +367,8 @@ static void g_singleton_factory_class_init(GSingletonFactoryClass *klass)  static void g_singleton_factory_init(GSingletonFactory *factory)  { -    factory->table = g_hash_table_new_full((GHashFunc)g_singleton_candidate_hash, -                                           (GEqualFunc)g_singleton_candidate_is_equal, +    factory->table = g_hash_table_new_full((GHashFunc)g_hashable_object_hash, +                                           (GEqualFunc)g_comparable_object_is_equal,                                             g_object_unref, NULL);      g_mutex_init(&factory->access); @@ -524,6 +527,8 @@ GSingletonCandidate *g_singleton_factory_get_instance(GSingletonFactory *factory          g_hash_table_add(factory->table, candidate);  #endif +        g_singleton_candidate_mark_as_read_only(candidate); +          result = candidate;      } diff --git a/src/glibext/singleton.h b/src/glibext/singleton.h index 36714fa..11afffd 100644 --- a/src/glibext/singleton.h +++ b/src/glibext/singleton.h @@ -25,6 +25,9 @@  #define _GLIBEXT_SINGLETON_H +#include <stdbool.h> + +  #include "helpers.h" @@ -37,14 +40,11 @@  DECLARE_INTERFACE(GSingletonCandidate, g_singleton_candidate, G, SINGLETON_CANDIDATE); -/* Fournit une liste de candidats embarqués par un candidat. */ -GSingletonCandidate **g_singleton_candidate_list_inner_instances(const GSingletonCandidate *, size_t *); - -/* Fournit l'empreinte d'un candidat à une centralisation. */ -guint g_singleton_candidate_hash(GSingletonCandidate *); +/* Indique si le candidat est figé. */ +bool g_singleton_candidate_is_read_only(const GSingletonCandidate *); -/* Détermine si deux candidats à l'unicité sont identiques. */ -gboolean g_singleton_candidate_is_equal(GSingletonCandidate *, GSingletonCandidate *); +/* Crée une copie modifiable d'un object unique. */ +GSingletonCandidate *g_singleton_candidate_dup(const GSingletonCandidate *); diff --git a/src/glibext/strbuilder-int.h b/src/glibext/strbuilder-int.h index 97e0f4e..d74e65c 100644 --- a/src/glibext/strbuilder-int.h +++ b/src/glibext/strbuilder-int.h @@ -2,7 +2,7 @@  /* Chrysalide - Outil d'analyse de fichiers binaires   * strbuilder-int.h - définitions internes propres aux éléments exportant une chaîne de caractères   * - * Copyright (C) 2024 Cyrille Bagard + * Copyright (C) 2025 Cyrille Bagard   *   *  This file is part of Chrysalide.   * diff --git a/src/glibext/strbuilder.c b/src/glibext/strbuilder.c index 54a102b..e48674c 100644 --- a/src/glibext/strbuilder.c +++ b/src/glibext/strbuilder.c @@ -2,7 +2,7 @@  /* Chrysalide - Outil d'analyse de fichiers binaires   * strbuilder.c - exportation d'une chaîne de caractères depuis un élément   * - * Copyright (C) 2024 Cyrille Bagard + * Copyright (C) 2025 Cyrille Bagard   *   *  This file is part of Chrysalide.   * @@ -51,6 +51,7 @@ G_DEFINE_INTERFACE(GStringBuilder, g_string_builder, G_TYPE_OBJECT)  static void g_string_builder_default_init(GStringBuilderInterface *iface)  { +    iface->to_string = NULL;  } @@ -77,7 +78,11 @@ bool g_string_builder_to_string(const GStringBuilder *builder, unsigned int flag      iface = G_STRING_BUILDER_GET_IFACE(builder); -    result = iface->to_string(builder, flags, out); +    if (iface->to_string == NULL) +        result = false; + +    else +        result = iface->to_string(builder, flags, out);      return result; diff --git a/src/glibext/strbuilder.h b/src/glibext/strbuilder.h index d8d0eaa..b6422f7 100644 --- a/src/glibext/strbuilder.h +++ b/src/glibext/strbuilder.h @@ -2,7 +2,7 @@  /* Chrysalide - Outil d'analyse de fichiers binaires   * strbuilder.h - prototypes pour l'exportation d'une chaîne de caractères depuis un élément   * - * Copyright (C) 2024 Cyrille Bagard + * Copyright (C) 2025 Cyrille Bagard   *   *  This file is part of Chrysalide.   * | 
