/* Chrysalide - Outil d'analyse de fichiers binaires * helpers.c - simplification des interactions de base avec Python * * Copyright (C) 2018-2024 Cyrille Bagard * * This file is part of Chrysalide. * * Chrysalide is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * Chrysalide is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with Chrysalide. If not, see . */ #include "helpers.h" #include #include #include #include #include #include #include #include #ifdef INCLUDE_GTK_SUPPORT # include #endif #include #include #include "access.h" #include "strenum.h" /* ---------------------------- MISE EN PLACE DE MODULES ---------------------------- */ /* Ajoute une classe dans les fonctionnalités globales. */ static bool include_python_type_into_features(PyObject *, PyTypeObject *); /* --------------------------- CONFORTS CIBLANT PYGOBJECT --------------------------- */ /* Message d'erreur affiché. */ #define NO_CONSTRUCTOR_MSG _("Chrysalide does not allow building this kind of object from Python") /* Message d'erreur affiché puis recherché. */ #define NOT_IMPLEMENTED_ROUTINE_MSG _("Chrysalide method implementation is missing") /* Message d'erreur affiché puis recherché. */ #define NOT_IMPLEMENTED_GETTER_MSG _("Chrysalide getter implementation is missing") /* Détermine une documentation adaptée à un type interne. */ static void define_auto_documentation(PyTypeObject *); /* ---------------------------------------------------------------------------------- */ /* ACCELERATEURS POUR PYTHON UNIQUEMENT */ /* ---------------------------------------------------------------------------------- */ /****************************************************************************** * * * Paramètres : status = bilan de comparaison à traduire. * * op = type de comparaison menée. * * * * Description : Traduit pour Python le bilan d'une comparaison riche. * * * * Retour : Objet Python à référencer. * * * * Remarques : - * * * ******************************************************************************/ PyObject *status_to_rich_cmp_state(int status, int op) { PyObject *result; /* Bilan àretourner */ switch (op) { case Py_LT: result = status < 0 ? Py_True : Py_False; break; case Py_LE: result = status <= 0 ? Py_True : Py_False; break; case Py_EQ: result = status == 0 ? Py_True : Py_False; break; case Py_NE: result = status != 0 ? Py_True : Py_False; break; case Py_GT: result = status > 0 ? Py_True : Py_False; break; case Py_GE: result = status >= 0 ? Py_True : Py_False; break; default: result = Py_NotImplemented; break; } return result; } /****************************************************************************** * * * Paramètres : arg = argument quelconque à tenter de convertir. * * dst = destination des valeurs récupérées en cas de succès. * * * * Description : Tente de convertir en élément appelable. * * * * Retour : Bilan de l'opération, voire indications supplémentaires. * * * * Remarques : - * * * ******************************************************************************/ int convert_to_callable(PyObject *arg, void *dst) { int result; /* Bilan à retourner */ result = PyCallable_Check(arg); switch (result) { case -1: /* L'exception est déjà fixée par Python */ result = 0; break; case 0: PyErr_SetString(PyExc_TypeError, "unable to convert the provided argument to a callable object"); break; case 1: *((PyObject **)dst) = arg; break; default: assert(false); break; } return result; } /****************************************************************************** * * * Paramètres : target = propriétaire de la routine visée. * * method = désignation de la fonction à appeler. * * * * Description : Indique si une routine Python existe ou non. * * * * Retour : Bilan de l'analyse. * * * * Remarques : - * * * ******************************************************************************/ bool has_python_method(PyObject *module, const char *method) { bool result; /* Bilan à retourner */ PyObject *func; /* Fonction visée */ result = (PyObject_HasAttrString(module, method) == 1); if (result) { func = PyObject_GetAttrString(module, method); assert(func != NULL); result = PyCallable_Check(func); Py_DECREF(func); } return result; } /****************************************************************************** * * * Paramètres : target = propriétaire de la routine visée. * * method = désignation de la fonction à appeler. * * * * Description : Indique si une routine Python possède une implémentation. * * * * Retour : Bilan de l'analyse. * * * * Remarques : - * * * ******************************************************************************/ bool has_python_implementation_method(PyObject *module, const char *method) { bool result; /* Bilan à retourner */ PyObject *func; /* Fonction visée */ const PyMethodDef *def; /* Définition de la fonction */ result = (PyObject_HasAttrString(module, method) == 1); if (result) { func = PyObject_GetAttrString(module, method); assert(func != NULL); result = PyCallable_Check(func); if (func->ob_type == &PyCFunction_Type) { def = ((PyCFunctionObject *)func)->m_ml; assert(strcmp(def->ml_name, method) == 0); result = (def->ml_meth != not_yet_implemented_method); } Py_DECREF(func); } return result; } /****************************************************************************** * * * Paramètres : target = propriétaire de la routine visée. * * method = désignation de la fonction à appeler. * * args = arguments à associer à l'opération. * * * * Description : Appelle une routine Python. * * * * Retour : Retour obtenu ou NULL si erreur. * * * * Remarques : - * * * ******************************************************************************/ PyObject *run_python_method(PyObject *module, const char *method, PyObject *args) { PyObject *result; /* Bilan à retourner */ PyObject *func; /* Fonction visée */ PyObject *type; /* Type d'exception levée */ PyObject *value; /* Détails particuliers */ PyObject *traceback; /* Pile d'appels de l'exception*/ PyObject *refmsg; /* Message de référence */ assert(PyGILState_Check() == 1); /* Exécution */ result = NULL; func = PyObject_GetAttrString(module, method); if (func == NULL) goto check_error; if (PyCallable_Check(func)) result = PyObject_CallObject(func, args); Py_DECREF(func); /* Répercutions */ check_error: PyErr_Fetch(&type, &value, &traceback); if (type != NULL && type == PyExc_NotImplementedError \ && value != NULL && PyUnicode_Check(value)) { refmsg = PyUnicode_FromString(NOT_IMPLEMENTED_ROUTINE_MSG); if (PyUnicode_Compare(value, refmsg) == 0) { Py_DECREF(value); value = PyUnicode_FromFormat(_("method implementation is missing for '%s'"), method); } Py_DECREF(refmsg); } PyErr_Restore(type, value, traceback); return result; } /* ---------------------------------------------------------------------------------- */ /* MISE EN PLACE DE MODULES */ /* ---------------------------------------------------------------------------------- */ /****************************************************************************** * * * Paramètres : super = module dont la définition est à compléter. * * def = définition du module à créer. * * * * Description : Met en place un nouveau module Python. * * * * Retour : Bilan de l'opération. * * * * Remarques : - * * * ******************************************************************************/ PyObject *build_python_module(PyObject *super, PyModuleDef *def) { PyObject *result; /* Création à retourner */ int ret; /* Bilan d'un appel */ #if PY_VERSION_HEX >= 0x03070000 PyObject *modules; /* Modules de l'interpréteur */ #endif char *dot; /* Dernier point de ce chemin */ result = PyModule_Create(def); if (result == NULL) goto quick_bad_exit; ret = PyState_AddModule(super, def); if (ret != 0) goto bad_exit; #if PY_VERSION_HEX >= 0x03070000 modules = PyImport_GetModuleDict(); ret = _PyImport_FixupBuiltin(result, def->m_name, modules); #else ret = _PyImport_FixupBuiltin(result, def->m_name); #endif if (ret != 0) goto bad_exit; dot = strrchr(def->m_name, '.'); assert(dot != NULL); Py_INCREF(result); ret = PyModule_AddObject(super, dot + 1, result); if (ret != 0) { Py_DECREF(result); goto bad_exit; } register_access_to_python_module(def->m_name, result); return result; bad_exit: Py_DECREF(result); quick_bad_exit: assert(false); return false; } /****************************************************************************** * * * Paramètres : module = module dont la définition est à compléter. * * defs = définitions de méthodes pour module. * * * * Description : Met en place une série de méthodes pour un module Python. * * * * Retour : Bilan de l'opération. * * * * Remarques : - * * * ******************************************************************************/ bool register_python_module_methods(PyObject *module, PyMethodDef *defs) { bool result; /* Bilan à retourner */ int ret; /* Bilan d'un appel */ PyMethodDef *iter; /* Boucle de parcours */ PyObject *features; /* Module à recompléter */ PyObject *features_dict; /* Dictionnaire à compléter */ PyObject *mod_dict; /* Dictionnaire à consulter */ PyObject *item; /* Nouvel élément à exporter */ ret = PyModule_AddFunctions(module, defs); result = (ret == 0); if (result) { features = get_access_to_python_module("pychrysalide.features"); features_dict = PyModule_GetDict(features); mod_dict = PyModule_GetDict(module); for (iter = defs; iter->ml_name != NULL && result; iter++) { item = PyDict_GetItemString(mod_dict, iter->ml_name); result = (item != NULL); assert(result); if (result) { Py_INCREF(item); ret = PyDict_SetItemString(features_dict, iter->ml_name, item); result = (ret == 0); assert(result); } } } return result; } /****************************************************************************** * * * Paramètres : dict = dictionnaire où conserver une référence au type créé. * * type = type dans sa version Python. * * * * Description : Ajoute une classe dans les fonctionnalités globales. * * * * Retour : Bilan de l'opération. * * * * Remarques : - * * * ******************************************************************************/ static bool include_python_type_into_features(PyObject *dict, PyTypeObject *type) { bool result; /* Bilan à retourner */ PyObject *features; /* Module à recompléter */ PyObject *features_dict; /* Dictionnaire à compléter */ char *name; /* Désignation de la classe */ PyObject *item; /* Nouvel élément à exporter */ int ret; /* Bilan d'une insertion */ features = get_access_to_python_module("pychrysalide.features"); features_dict = PyModule_GetDict(features); name = strrchr(type->tp_name, '.'); assert(name != NULL); name++; item = PyDict_GetItemString(dict, name); result = (item != NULL); assert(result); Py_INCREF(item); ret = PyDict_SetItemString(features_dict, name, item); result = (ret == 0); assert(result); return result; } /****************************************************************************** * * * Paramètres : module = module dont la définition est à compléter. * * type = type à intégrer dans sa version Python. * * * * Description : Met en place un objet au sein d'un module Python. * * * * Retour : Bilan de l'opération. * * * * Remarques : - * * * ******************************************************************************/ bool register_python_module_object(PyObject *module, PyTypeObject *type) { bool result; /* Bilan à retourner */ char *name; /* Désignation de la classe */ int ret; /* Bilan d'un appel */ PyObject *dict; /* Dictionnaire du module */ name = strrchr(type->tp_name, '.'); assert(name != NULL); name++; Py_INCREF(type); ret = PyModule_AddObject(module, name, (PyObject *)type); result = (ret == 0); if (!result) Py_DECREF(type); else { dict = PyModule_GetDict(module); result = include_python_type_into_features(dict, type); } return result; } /* ---------------------------------------------------------------------------------- */ /* CONFORTS CIBLANT PYGOBJECT */ /* ---------------------------------------------------------------------------------- */ /****************************************************************************** * * * Paramètres : type = type du nouvel objet à mettre en place. * * gbase = type de base natif. * * cinit = procédure d'initialisation de la classe associée. * * args = éventuelle liste d'arguments. * * kwds = éventuel dictionnaire de valeurs mises à disposition.* * * * Description : Accompagne la création d'une instance dérivée en Python. * * * * Retour : Nouvel objet Python mis en place ou NULL en cas d'échec. * * * * Remarques : - * * * ******************************************************************************/ PyObject *python_abstract_constructor(PyTypeObject *type, GType gbase, GClassInitFunc cinit, PyObject *args, PyObject *kwds) { PyObject *result; /* Objet à retourner */ PyTypeObject *base; /* Type parent version Python */ /* Validations diverses */ base = pygobject_lookup_class(gbase); if (type == base) { result = NULL; PyErr_Format(PyExc_RuntimeError, _("%s is an abstract class"), type->tp_name); goto exit; } /* On crée, et on laisse ensuite la main à PyGObject_Type.tp_init() */ result = PyType_GenericNew(type, args, kwds); exit: return result; } /****************************************************************************** * * * Paramètres : type = type du nouvel objet à mettre en place. * * args = éventuelle liste d'arguments. * * kwds = éventuel dictionnaire de valeurs mises à disposition. * * * * Description : Marque l'interdiction d'une instanciation depuis Python. * * * * Retour : NULL pour la levée d'exception. * * * * Remarques : - * * * ******************************************************************************/ PyObject *no_python_constructor_allowed(PyTypeObject *type, PyObject *args, PyObject *kwds) { PyObject *result; /* Exception à retourner */ result = NULL; PyErr_SetString(PyExc_NotImplementedError, NO_CONSTRUCTOR_MSG); return result; } /****************************************************************************** * * * Paramètres : self = objet quelconque dont le code Python hérite. * * args = série d'arguments si présents. * * * * Description : Marque l'absence d'implémentation pour une méthode donnée. * * * * Retour : NULL pour la levée d'exception. * * * * Remarques : - * * * ******************************************************************************/ PyObject *not_yet_implemented_method(PyObject *self, PyObject *args) { PyObject *result; /* Exception à retourner */ result = NULL; PyErr_SetString(PyExc_NotImplementedError, NOT_IMPLEMENTED_ROUTINE_MSG); return result; } /****************************************************************************** * * * Paramètres : self = objet quelconque. * * args = arguments fournis à l'appel. * * * * Description : Retourne toujours rien. * * * * Retour : None. * * * * Remarques : - * * * ******************************************************************************/ PyObject *py_return_none(PyObject *self, PyObject *args) { PyObject *result; /* Bilan à retourner */ result = Py_None; Py_INCREF(result); return result; } /****************************************************************************** * * * Paramètres : self = objet quelconque. * * args = arguments fournis à l'appel. * * * * Description : Retourne toujours faux. * * * * Retour : False. * * * * Remarques : - * * * ******************************************************************************/ PyObject *py_return_false(PyObject *self, PyObject *args) { PyObject *result; /* Bilan à retourner */ result = Py_False; Py_INCREF(result); return result; } /****************************************************************************** * * * Paramètres : self = objet quelconque. * * args = arguments fournis à l'appel. * * * * Description : Retourne toujours vrai. * * * * Retour : False. * * * * Remarques : - * * * ******************************************************************************/ PyObject *py_return_true(PyObject *self, PyObject *args) { PyObject *result; /* Bilan à retourner */ result = Py_True; Py_INCREF(result); return result; } /****************************************************************************** * * * Paramètres : self = objet quelconque dont le code Python hérite. * * closure = non utilisé ici. * * * * Description : Marque l'absence d'implémentation pour un attribut donné. * * * * Retour : NULL pour la levée d'exception. * * * * Remarques : - * * * ******************************************************************************/ PyObject *not_yet_implemented_getter(PyObject *self, void *closure) { PyObject *result; /* Exception à retourner */ result = NULL; PyErr_SetString(PyExc_NotImplementedError, NOT_IMPLEMENTED_GETTER_MSG); return result; } /****************************************************************************** * * * Paramètres : otype = définition à mettre en place dynamiquement. * * * * Description : Définit dans le tas de Python un nouveau type. * * * * Retour : Nouveau type prêt à emploi. * * * * Remarques : - * * * ******************************************************************************/ PyTypeObject *define_python_dynamic_type(const PyTypeObject *otype) { PyTypeObject *result; /* Définition créée à renvoyer */ PyType_Slot slots[10]; /* Emplacements pour infos */ PyType_Spec spec; /* Définition du type */ PyType_Slot *iter; /* Boucle de parcours */ PyObject *bases; /* Bases de construction */ bases = PyTuple_Pack(1, &PyType_Type); spec.name = otype->tp_name; spec.basicsize = otype->tp_basicsize; spec.flags = otype->tp_flags; spec.slots = slots; iter = &slots[0]; if (otype->tp_doc != NULL) { iter->slot = Py_tp_doc; iter->pfunc = (void *)otype->tp_doc; iter++; } if (otype->tp_methods != NULL) { iter->slot = Py_tp_methods; iter->pfunc = otype->tp_methods; iter++; } if (otype->tp_getset != NULL) { iter->slot = Py_tp_getset; iter->pfunc = otype->tp_getset; iter++; } if (otype->tp_init != NULL) { iter->slot = Py_tp_init; iter->pfunc = otype->tp_init; iter++; } if (otype->tp_new != NULL) { iter->slot = Py_tp_new; iter->pfunc = otype->tp_new; iter++; } iter->slot = 0; result = (PyTypeObject *)PyType_FromSpecWithBases(&spec, bases); Py_DECREF(bases); return result; } /****************************************************************************** * * * Paramètres : type = type dans sa version Python. * * * * Description : Détermine une documentation adaptée à un type interne. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void define_auto_documentation(PyTypeObject *type) { /** * L'idée est ici d'éviter la documentation automatique générée par * pyg_object_descr_doc_get(). */ PyDict_SetItemString(type->tp_dict, "__doc__", PyUnicode_FromString(type->tp_doc)); } /****************************************************************************** * * * Paramètres : dict = dictionnaire où conserver une référence au type créé.* * gtype = type dans sa version GLib. * * type = type dans sa version Python. * * * * Description : Enregistre correctement une surcouche de conversion GObject. * * * * Retour : Bilan de l'opération. * * * * Remarques : - * * * ******************************************************************************/ bool register_class_for_pygobject(PyObject *dict, GType gtype, PyTypeObject *type) { bool result; /* Bilan à retourner */ GType parent_type; /* Type parent version GObject */ PyTypeObject *base; /* Type parent version Python */ assert(gtype != G_TYPE_INVALID); /** * pygobject_register_class() définit type->tp_base à partir des arguments fournis, * puis fait appel à PyType_Ready(). * * PyType_Ready() complète la définition via inherit_special() : * * type->tp_basicsize = type->tp_base->tp_basicsize * * Cependant, il y a un appel à mro_internal() avant, qui mène à solid_base() * puis à extra_ivars(). Et là : * * size_t t_size = type->tp_basicsize; * size_t b_size = base->tp_basicsize; * * assert(t_size >= b_size); * * Si le type de base est spécifié, une taille doit être indiquée. * * Et quelqu'un doit se coller à la tâche. PyGObject ne fait rien, donc... */ parent_type = g_type_parent(gtype); base = pygobject_lookup_class(parent_type); if (type->tp_basicsize < base->tp_basicsize) { assert(type->tp_basicsize == 0); type->tp_basicsize = base->tp_basicsize; } pygobject_register_class(dict, NULL, gtype, type, NULL); if (PyErr_Occurred() == NULL) result = true; else { PyErr_Print(); result = false; } assert(PyErr_Occurred() == NULL); /** * Création d'un dictionnaire complet pour la simulation d'un "import *". */ if (result && startswith(type->tp_name, "pychrysalide.")) { define_auto_documentation(type); result = include_python_type_into_features(dict, type); } return result; } /****************************************************************************** * * * Paramètres : dict = dictionnaire où conserver une référence au type créé.* * gtype = type dans sa version GLib. * * type = type dans sa version Python. * * * * Description : Enregistre correctement une interface GObject pour Python. * * * * Retour : Bilan de l'opération. * * * * Remarques : - * * * ******************************************************************************/ bool register_interface_for_pygobject(PyObject *dict, GType gtype, PyTypeObject *type, const GInterfaceInfo *info) { bool result; /* Bilan à retourner */ char *name; /* Désignation de la classe */ assert(gtype != G_TYPE_INVALID); name = strrchr(type->tp_name, '.'); assert(name != NULL); name++; pyg_register_interface(dict, name, gtype, type); pyg_register_interface_info(gtype, info); if (startswith(type->tp_name, "pychrysalide.")) { define_auto_documentation(type); result = include_python_type_into_features(dict, type); } else result = true; return result; } /****************************************************************************** * * * Paramètres : gtype = type dans sa version GLib. * * type = type dans sa version Python. * * * * Description : Enregistre un type Python dérivant d'un type GLib dynamique. * * * * Retour : Bilan de l'opération. * * * * Remarques : - * * * ******************************************************************************/ bool register_class_for_dynamic_pygobject(GType gtype, PyTypeObject *type) { bool result; /* Bilan à retourner */ PyTypeObject *legacy_parent; /* Type parent d'origine */ PyObject *sys_mod_dict; /* Dictionnaire des modules */ PyObject *modname; /* Nom du module du type */ PyObject *module; /* Module à recompléter */ PyObject *dict; /* Dictionnaire dudit module */ Py_ssize_t i; /* Boucle de parcours */ PyObject *mro_base; /* Base finale effective */ GType itype; /* Type d'interface implémentée*/ GQuark pyginterface_info_key; /* Clef d'accès non déclarée */ const GInterfaceInfo *iinfo; /* Informations associées */ /** * Lors de l'appel à pygobject_register_class(), PyGObject remplace systématiquement * le type Python fourni par : * * Py_TYPE(type) = PyGObject_MetaType; * * Ce nouveau type est le suivant (cf. gi/types.py) : * * class _GObjectMetaBase(type): * """Metaclass for automatically registering GObject classes.""" * * D'une part, comme les enregistrements sont gérés manuellement dans le cas de * types dérivés dynamiquement d'objets natifs, ce changement est inutile. * * D'autre part, il semble avoir un soucis de références (cf. testGarbageCollecting() * du fichier de test plugins/plugin.py) : * * python3dm: ../Modules/gcmodule.c:379: visit_decref: Assertion `_PyGCHead_REFS(gc) != 0' failed. * * #3 __GI___assert_fail () * #4 visit_decref () * #5 subtype_traverse () * #6 subtract_refs () * #7 collect () * #8 collect_with_callback () * #9 gc_collect () * * On restaure donc le type d'origine de l'objet Python créé dynamquement pour éviter * ce genre de soucis. */ legacy_parent = Py_TYPE(type); sys_mod_dict = PyImport_GetModuleDict(); modname = PyDict_GetItemString(type->tp_dict, "__module__"); module = PyObject_GetItem(sys_mod_dict, modname); dict = PyModule_GetDict(module); result = register_class_for_pygobject(dict, gtype, type); Py_SET_TYPE(type, legacy_parent); /** * Comme la mise en place dynamique de nouveau GType court-circuite les * mécanismes internes de pygobject, les interfaces implémentées ne sont * nominalement plus complétées. * * On reprend donc la logique copiée depuis le contenu de la fonction * pyg_type_add_interfaces() du fichier "pygobject-3.22.0/gi/gobjectmodule.c". */ for (i = 0; i < PyTuple_GET_SIZE(type->tp_mro); i++) { mro_base = PyTuple_GET_ITEM(type->tp_mro, i); if (!PyType_Check(mro_base)) continue; itype = pyg_type_from_object(mro_base); if (itype == G_TYPE_INTERFACE) continue; if (!G_TYPE_IS_INTERFACE(itype)) continue; if (!g_type_is_a(gtype, itype)) { /** * Reproduction de pyg_lookup_interface_info(). */ pyginterface_info_key = g_quark_from_static_string("PyGInterface::info"); iinfo = g_type_get_qdata(itype, pyginterface_info_key); assert(iinfo != NULL); g_type_add_interface_static(gtype, itype, iinfo); } } return result; } /****************************************************************************** * * * Paramètres : self = objet Python/GObject à initialiser. * * * * Description : Fait suivre à la partie GObject une initialisation nouvelle. * * * * Retour : Bilan de l'opération. * * * * Remarques : - * * * ******************************************************************************/ int forward_pygobjet_init(PyObject *self) { bool result; /* Bilan à retourner */ PyObject *new_args; /* Nouveaux arguments épurés */ PyObject *new_kwds; /* Nouveau dictionnaire épuré */ new_args = PyTuple_New(0); new_kwds = PyDict_New(); result = PyGObject_Type.tp_init(self, new_args, new_kwds); Py_DECREF(new_kwds); Py_DECREF(new_args); return result; } /****************************************************************************** * * * Paramètres : arg = argument quelconque à tenter de convertir. * * dst = destination des valeurs récupérées en cas de succès. * * * * Description : Tente de convertir en valeur GType. * * * * Retour : Bilan de l'opération, voire indications supplémentaires. * * * * Remarques : - * * * ******************************************************************************/ int convert_to_gtype(PyObject *arg, void *dst) { int result; /* Bilan à retourner */ GType type; /* Type obtenu ou 0 */ type = pyg_type_from_object(arg); switch (type) { case G_TYPE_INVALID: /* L'exception est déjà fixée par Python */ result = 0; break; default: *((GType *)dst) = type; result = 1; break; } return result; } /****************************************************************************** * * * Paramètres : arg = argument quelconque à tenter de convertir. * * dst = destination des valeurs récupérées en cas de succès. * * * * Description : Tente de convertir en instance GObject. * * * * Retour : Bilan de l'opération, voire indications supplémentaires. * * * * Remarques : - * * * ******************************************************************************/ int convert_to_gobject(PyObject *arg, void *dst) { int result; /* Bilan à retourner */ result = PyObject_IsInstance(arg, (PyObject *)&PyGObject_Type); switch (result) { case -1: /* L'exception est déjà fixée par Python */ result = 0; break; case 0: PyErr_SetString(PyExc_TypeError, "unable to convert the provided argument to GObject instance"); break; case 1: *((GObject **)dst) = G_OBJECT(pygobject_get(arg)); break; default: assert(false); break; } return result; } #if 0 #ifdef INCLUDE_GTK_SUPPORT /****************************************************************************** * * * Paramètres : arg = argument quelconque à tenter de convertir. * * dst = destination des valeurs récupérées en cas de succès. * * * * Description : Tente de convertir en instance de composant GTK. * * * * Retour : Bilan de l'opération, voire indications supplémentaires. * * * * Remarques : - * * * ******************************************************************************/ int convert_to_gtk_widget(PyObject *arg, void *dst) { int result; /* Bilan à retourner */ PyObject *gtk_mod; /* Module Python Gtk */ PyObject *widget_type; /* Module "GtkWidget" */ int ret; /* Bilan d'une conversion */ result = 0; gtk_mod = PyImport_ImportModule("gi.repository.Gtk"); if (gtk_mod == NULL) { PyErr_SetString(PyExc_TypeError, "unable to find the Gtk Python module"); goto done; } widget_type = PyObject_GetAttrString(gtk_mod, "Widget"); Py_DECREF(gtk_mod); ret = PyObject_TypeCheck(arg, (PyTypeObject *)widget_type); Py_DECREF(widget_type); if (!ret) { PyErr_SetString(PyExc_TypeError, "unable to convert the provided argument to GTK widget"); goto done; } *((GtkWidget **)dst) = GTK_WIDGET(pygobject_get(arg)); result = 1; done: return result; } /****************************************************************************** * * * Paramètres : arg = argument quelconque à tenter de convertir. * * dst = destination des valeurs récupérées en cas de succès. * * * * Description : Tente de convertir en instance de conteneur GTK. * * * * Retour : Bilan de l'opération, voire indications supplémentaires. * * * * Remarques : - * * * ******************************************************************************/ int convert_to_gtk_container(PyObject *arg, void *dst) { int result; /* Bilan à retourner */ PyObject *gtk_mod; /* Module Python Gtk */ PyObject *container_type; /* Module "GtkContainer" */ int ret; /* Bilan d'une conversion */ result = 0; gtk_mod = PyImport_ImportModule("gi.repository.Gtk"); if (gtk_mod == NULL) { PyErr_SetString(PyExc_TypeError, "unable to find the Gtk Python module"); goto done; } container_type = PyObject_GetAttrString(gtk_mod, "Container"); Py_DECREF(gtk_mod); ret = PyObject_TypeCheck(arg, (PyTypeObject *)container_type); Py_DECREF(container_type); if (!ret) { PyErr_SetString(PyExc_TypeError, "unable to convert the provided argument to GTK container"); goto done; } *((GtkContainer **)dst) = GTK_CONTAINER(pygobject_get(arg)); result = 1; done: return result; } #endif /****************************************************************************** * * * Paramètres : color = couleur dans sa définition native à copier. * * * * Description : Construit un objet Python pour une couleur RGBA. * * * * Retour : Objet Python prêt à emploi ou NULL en cas d'échec. * * * * Remarques : - * * * ******************************************************************************/ PyObject *create_gdk_rgba(const GdkRGBA *color) { PyObject *result; /* Coloration à retourner */ PyObject *gdk_mod; /* Module Python Gdk */ PyObject *rgba_type; /* Classe "GtkRGBA" */ PyObject *rgba_args; /* Arguments pour l'appel */ result = NULL; gdk_mod = PyImport_ImportModule("gi.repository.Gdk"); if (gdk_mod == NULL) { PyErr_SetString(PyExc_TypeError, "unable to find the Gtk Python module"); goto done; } rgba_type = PyObject_GetAttrString(gdk_mod, "RGBA"); Py_DECREF(gdk_mod); rgba_args = PyTuple_New(4); PyTuple_SetItem(rgba_args, 0, PyFloat_FromDouble(color->red)); PyTuple_SetItem(rgba_args, 1, PyFloat_FromDouble(color->green)); PyTuple_SetItem(rgba_args, 2, PyFloat_FromDouble(color->blue)); PyTuple_SetItem(rgba_args, 3, PyFloat_FromDouble(color->alpha)); result = PyObject_CallObject(rgba_type, rgba_args); Py_DECREF(rgba_args); done: return result; } /****************************************************************************** * * * Paramètres : arg = argument quelconque à tenter de convertir. * * dst = destination des valeurs récupérées en cas de succès. * * * * Description : Tente de convertir en instance de couleur RGBA. * * * * Retour : Bilan de l'opération, voire indications supplémentaires. * * * * Remarques : - * * * ******************************************************************************/ int convert_to_gdk_rgba(PyObject *arg, void *dst) { int result; /* Bilan à retourner */ PyObject *gdk_mod; /* Module Python Gdk */ PyObject *rgba_type; /* Module "RGBA" */ int ret; /* Bilan d'une conversion */ PyObject *value; /* Valeur d'une composante */ result = 0; gdk_mod = PyImport_ImportModule("gi.repository.Gdk"); if (gdk_mod == NULL) { PyErr_SetString(PyExc_TypeError, "unable to find the Gdk Python module"); goto done; } rgba_type = PyObject_GetAttrString(gdk_mod, "RGBA"); Py_DECREF(gdk_mod); ret = PyObject_TypeCheck(arg, (PyTypeObject *)rgba_type); Py_DECREF(rgba_type); if (!ret) { PyErr_SetString(PyExc_TypeError, "unable to convert the provided argument to GDK RGBA color"); goto done; } value = PyObject_GetAttrString(arg, "red"); assert(PyFloat_Check(value)); ((GdkRGBA *)dst)->red = PyFloat_AsDouble(value); value = PyObject_GetAttrString(arg, "blue"); assert(PyFloat_Check(value)); ((GdkRGBA *)dst)->blue = PyFloat_AsDouble(value); value = PyObject_GetAttrString(arg, "green"); assert(PyFloat_Check(value)); ((GdkRGBA *)dst)->green = PyFloat_AsDouble(value); value = PyObject_GetAttrString(arg, "alpha"); assert(PyFloat_Check(value)); ((GdkRGBA *)dst)->alpha = PyFloat_AsDouble(value); result = 1; done: return result; } #endif /****************************************************************************** * * * Paramètres : arg = argument quelconque à tenter de convertir. * * dst = destination des valeurs récupérées en cas de succès. * * * * Description : Tente de convertir en tableau de chaînes de caractères. * * * * Retour : Bilan de l'opération, voire indications supplémentaires. * * * * Remarques : - * * * ******************************************************************************/ int convert_to_sequence_to_charp_array(PyObject *arg, void *dst) { int result; /* Bilan à retourner */ charp_array_t *array; /* Tableau à constituer */ size_t i; /* Boucle de parcours */ PyObject *value; /* Valeur brute d'un élément */ array = (charp_array_t *)dst; /* Nettoyage ? */ if (arg == NULL) { result = 1; goto clean; } else { result = 0; if (PySequence_Check(arg) != 1) goto done; array->length = PySequence_Length(arg); array->values = calloc(array->length, sizeof(char *)); for (i = 0; i < array->length; i++) { value = PySequence_ITEM(arg, i); if (!PyUnicode_Check(value)) { Py_DECREF(value); goto clean; } array->values[i] = strdup(PyUnicode_DATA(value)); Py_DECREF(value); } result = Py_CLEANUP_SUPPORTED; } done: return result; clean: clean_charp_array(array); return result; } /****************************************************************************** * * * Paramètres : array = tableau de chaînes de caractères à traiter. * * * * Description : Libère de la mémoire un tableau de chaînes de caractères. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void clean_charp_array(charp_array_t *array) { size_t i; /* Boucle de parcours */ for (i = 0; i < array->length; i++) if (array->values[i] != NULL) free(array->values[i]); if (array->values != NULL) free(array->values); array->values = NULL; array->length = 0; } /* ---------------------------------------------------------------------------------- */ /* TRANSFERT DES VALEURS CONSTANTES */ /* ---------------------------------------------------------------------------------- */ /****************************************************************************** * * * Paramètres : owner = désignation du propriétaire du dictionnaire visé. * * dict = dictionnaire dont le contenu est à compléter. * * flags = indique le type d'énumération ciblée. * * name = désignation humaine du groupe à constituer. * * values = noms et valeurs associées. * * doc = documentation à associer au groupe. * * * * Description : Officialise un groupe de constantes avec sémentique. * * * * Retour : Groupe de constantes mis en place ou NULL en cas d'échec. * * * * Remarques : - * * * ******************************************************************************/ PyObject *_attach_constants_group(const char *owner, PyObject *dict, bool flags, const char *name, PyObject *values, const char *doc) { PyObject *result; /* Instance à retourner */ PyObject *enum_mod; /* Module Python enum */ PyObject *class; /* Classe "Enum*" */ PyObject *str_obj; /* Conversion en Python */ int ret; /* Bilan d'une insertion */ PyObject *args; /* Arguments de la construction*/ PyObject *kwargs; /* Mots clefs en complément */ char *dot; /* Point de séparation */ char *module; /* Module d'appartenance */ char *qualname; /* Désignation pour Pickle */ PyObject *new; /* Nouvelle instance en place */ PyObject *features; /* Module à recompléter */ PyObject *features_dict; /* Dictionnaire à compléter */ result = NULL; /* Recherche de la classe Python */ enum_mod = PyImport_ImportModule("enum"); if (enum_mod == NULL) goto no_mod; if (flags) class = PyObject_GetAttrString(enum_mod, "IntFlag"); else class = PyObject_GetAttrString(enum_mod, "IntEnum"); Py_DECREF(enum_mod); if (class == NULL) goto no_class; /* Compléments des paramètres */ str_obj = PyUnicode_FromString(doc); ret = PyDict_SetItemString(values, "__doc__", str_obj); Py_DECREF(str_obj); if (ret != 0) goto doc_error; args = PyTuple_New(2); ret = PyTuple_SetItem(args, 0, PyUnicode_FromString(name)); if (ret != 0) goto args_error; Py_INCREF(values); ret = PyTuple_SetItem(args, 1, values); if (ret != 0) goto args_error; kwargs = PyDict_New(); dot = rindex(owner, '.'); if (dot == NULL) { str_obj = PyUnicode_FromString(owner); ret = PyDict_SetItemString(kwargs, "module", str_obj); Py_DECREF(str_obj); if (ret != 0) goto kwargs_error; str_obj = PyUnicode_FromString(name); ret = PyDict_SetItemString(kwargs, "qualname", str_obj); Py_DECREF(str_obj); if (ret != 0) goto kwargs_error; } else { module = strndup(owner, dot - owner); str_obj = PyUnicode_FromString(module); ret = PyDict_SetItemString(kwargs, "module", str_obj); Py_DECREF(str_obj); free(module); if (ret != 0) goto kwargs_error; asprintf(&qualname, "%s.%s", dot + 1, name); str_obj = PyUnicode_FromString(qualname); ret = PyDict_SetItemString(kwargs, "qualname", str_obj); Py_DECREF(str_obj); free(qualname); if (ret != 0) goto kwargs_error; } /* Constitution de l'énumération et enregistrement */ new = PyObject_Call(class, args, kwargs); if (new == NULL) goto build_error; ret = PyDict_SetItemString(dict, name, new); if (ret != 0) goto register_0_error; features = get_access_to_python_module("pychrysalide.features"); features_dict = PyModule_GetDict(features); ret = PyDict_SetItemString(features_dict, name, new); if (ret != 0) goto register_1_error; result = new; Py_INCREF(result); /* Sortie propre */ register_1_error: register_0_error: Py_DECREF(new); build_error: kwargs_error: Py_DECREF(kwargs); args_error: Py_DECREF(args); doc_error: Py_DECREF(class); no_class: no_mod: Py_DECREF(values); return result; } /****************************************************************************** * * * Paramètres : owner = désignation du propriétaire du dictionnaire visé. * * dict = dictionnaire dont le contenu est à compléter. * * flags = indique le type d'énumération ciblée. * * name = désignation humaine du groupe à constituer. * * values = noms et valeurs associées. * * doc = documentation à associer au groupe. * * gtype = énumération GLib à lier. * * * * Description : Officialise un groupe de constantes avec lien GLib. * * * * Retour : Groupe de constantes mis en place ou NULL en cas d'échec. * * * * Remarques : - * * * ******************************************************************************/ PyObject *_attach_constants_group_with_pyg_enum(const char *owner, PyObject *dict, bool flags, const char *name, PyObject *values, const char *doc, GType gtype) { PyObject *result; /* Instance à retourner */ PyObject *values_set; /* Zone pour nouvelles valeurs */ int ret; /* Bilan d'une insertion */ PyObject *new; /* Nouvelle instance en place */ PyObject *values_src; /* Source de nouvelles valeurs */ PyObject *values_dest; /* Destination des valeurs */ static GQuark pygenum_class_key = 0; /* Clef d'accès au marquage */ result = NULL; /** * Le seul intérêt d'un tel enregistrement en bonne et due forme est de * permettre une impression, via str() ou repr(), de l'énumération * transcrite en GLib via g_enum_register_static() et potentiellement * convertie de façon brusque par _pygi_argument_to_object(), lors d'une * émission de signal par exemple. * * La satisfaction de la fonction pyg_enum_from_gtype() est ainsi recherchée. * Tous les éléments sont normalement mis en place à partir de la fonction * pyg_enum_add(). */ /* Préparation du réceptacle */ values_set = PyDict_New(); ret = PyDict_SetItemString(values, "__enum_values__", values_set); Py_DECREF(values_set); if (ret != 0) goto exit; /* Création */ new = _attach_constants_group(owner, dict, flags, name, values, doc); if (new == NULL) goto exit; /* Actualisation des valeurs */ values_src = PyDict_GetItemString(((PyTypeObject *)new)->tp_dict, "_value2member_map_"); if (values_src == NULL) goto exit_without_src; values_dest = PyDict_GetItemString(((PyTypeObject *)new)->tp_dict, "__enum_values__"); if (values_dest == NULL) goto exit_without_dest; assert(values_dest == values_set); ret = PyDict_Merge(values_dest, values_src, true); if (ret == 0) { result = new; Py_INCREF(result); if (pygenum_class_key == 0) pygenum_class_key = g_quark_from_static_string("PyGEnum::class"); g_type_set_qdata(gtype, pygenum_class_key, result); } exit_without_dest: exit_without_src: Py_DECREF(new); exit: return result; } /****************************************************************************** * * * Paramètres : owner = désignation du propriétaire du dictionnaire visé. * * name = désignation humaine du groupe à consulter. * * value = valeur à transmettre à Python. * * * * Description : Traduit une valeur constante C en équivalent Python. * * * * Retour : Objet Python résultant ou NULL en cas d'erreur. * * * * Remarques : - * * * ******************************************************************************/ PyObject *_cast_with_constants_group(const char *owner, const char *name, unsigned long value) { PyObject *result; /* Objet Python à retourner */ char *dot; /* Position du dernier point */ char *modname; /* Chemin d'accès au module */ PyObject *module; /* Module à consulter */ PyObject *type; /* Classe propriétaire */ PyObject *class; /* Classe "Enum*" */ PyObject *args; /* Arguments de la construction*/ result = NULL; /* Recherche de la classe Python */ dot = strrchr(owner, '.'); if (dot == NULL) { module = get_access_to_python_module(owner); if (module == NULL) goto no_mod; type = module; Py_INCREF(type); } else { modname = strndup(owner, dot - owner); module = get_access_to_python_module(modname); free(modname); if (module == NULL) goto no_mod; type = PyObject_GetAttrString(module, dot + 1); if (type == NULL) goto no_type; } class = PyObject_GetAttrString(type, name); if (class == NULL) goto no_class; /* Construction */ args = Py_BuildValue("(k)", value); result = PyObject_CallObject(class, args); Py_DECREF(args); Py_DECREF(class); no_class: Py_DECREF(type); no_type: no_mod: return result; } /****************************************************************************** * * * Paramètres : dict = dictionnaire dont le contenu est à compléter. * * name = désignation humaine du groupe à constituer. * * doc = documentation à associer au groupe. * * out = dictionnaire à compléter. [OUT] * * * * Description : Officialise un groupe de constantes de chaînes de caractères.* * * * Retour : true en cas de succès de l'opération, false sinon. * * * * Remarques : - * * * ******************************************************************************/ bool _create_string_constants_group(PyObject *dict, const char *name, const char *doc, PyObject **out) { bool result; /* Bilan à retourner */ PyObject *class; /* Classe "Enum*" */ PyObject *args; /* Argument de construction */ int ret; /* Bilan d'une insertion */ PyObject *features; /* Module à recompléter */ PyObject *features_dict; /* Dictionnaire à compléter */ result = false; /* Recherche et instanciation de la classe Python */ class = (PyObject *)get_python_string_enum_type(); args = Py_BuildValue("(s)", doc); *out = PyObject_CallObject(class, args); Py_DECREF(args); if (*out == NULL) goto exit; /* Constitution de l'énumération et enregistrement */ ret = PyDict_SetItemString(dict, name, *out); if (ret != 0) goto register_0_error; features = get_access_to_python_module("pychrysalide.features"); features_dict = PyModule_GetDict(features); ret = PyDict_SetItemString(features_dict, name, *out); if (ret != 0) goto register_1_error; result = true; /* Sortie propre */ register_1_error: register_0_error: if (!result) Py_DECREF(*out); exit: return result; }