/* Chrysalide - Outil d'analyse de fichiers binaires * instruction.c - équivalent Python du fichier "arch/instruction.h" * * Copyright (C) 2018-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 this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "instruction.h" #include #include #include #include #include #include #include "constants.h" #include "operand.h" #include "vmpa.h" #include "../access.h" #include "../helpers.h" #include "../glibext/objhole.h" #include "../glibext/serialize.h" /* ------------------------ GLUE POUR CREATION DEPUIS PYTHON ------------------------ */ /* Initialise la classe générique des instructions. */ static int py_arch_instruction_init_gclass(GArchInstructionClass *, PyTypeObject *); CREATE_DYN_ABSTRACT_CONSTRUCTOR(arch_instruction, G_TYPE_ARCH_INSTRUCTION); /* Initialise une instance sur la base du dérivé de GObject. */ static int py_arch_instruction_init(PyObject *, PyObject *, PyObject *); /* Indique l'encodage d'une instruction de façon détaillée. */ static char *py_arch_instruction_get_encoding_wrapper(const GArchInstruction *); /* Fournit le nom humain de l'instruction manipulée. */ static char *py_arch_instruction_get_keyword_wrapper(const GArchInstruction *); /* ------------------- DEFINITION DES LIAISONS ENTRE INSTRUCTIONS ------------------- */ /* Fournit les origines d'une instruction donnée. */ static PyObject *py_arch_instruction_get_sources(PyObject *, void *); /* Fournit les destinations d'une instruction donnée. */ static PyObject *py_arch_instruction_get_destinations(PyObject *, void *); /* --------------------------- MANIPULATION DES OPERANDES --------------------------- */ /* Attache un opérande supplémentaire à une instruction. */ static PyObject *py_arch_instruction_attach_operand(PyObject *, PyObject *); /* Remplace un opérande d'une instruction par un autre. */ static PyObject *py_arch_instruction_replace_operand(PyObject *, PyObject *); /* Détache un opérande liée d'une instruction. */ static PyObject *py_arch_instruction_detach_operand(PyObject *, PyObject *); /* Détermine le chemin conduisant à un opérande. */ static PyObject *py_arch_instruction_find_operand_path(PyObject *, PyObject *); /* Obtient l'opérande correspondant à un chemin donné. */ static PyObject *py_arch_instruction_get_operand_from_path(PyObject *, PyObject *); /* Fournit tous les opérandes d'une instruction. */ static PyObject *py_arch_instruction_get_operands(PyObject *, void *); /* ------------------ LIAISON DE FONCTIONNALITES AVEC L'API PYTHON ------------------ */ /* Ajoute une information complémentaire à une instruction. */ static PyObject *py_arch_instruction_set_flag(PyObject *, PyObject *); /* Retire une information complémentaire à une instruction. */ static PyObject *py_arch_instruction_unset_flag(PyObject *, PyObject *); /* Détermine si une instruction possède un fanion particulier. */ static PyObject *py_arch_instruction_has_flag(PyObject *, PyObject *); /* Fournit l'identifiant correspondant à un type d'instructions. */ static PyObject *py_arch_instruction_get_type_id(PyObject *, void *); /* Indique l'encodage d'une instruction de façon détaillée. */ static PyObject *py_arch_instruction_get_encoding(PyObject *, void *); /* Indique l'encodage d'une instruction de façon détaillée. */ static PyObject *py_arch_instruction_get_keyword(PyObject *, void *); /* Fournit la place mémoire d'une instruction. */ static PyObject *py_arch_instruction_get_range(PyObject *, void *); /* ---------------------------------------------------------------------------------- */ /* GLUE POUR CREATION DEPUIS PYTHON */ /* ---------------------------------------------------------------------------------- */ /****************************************************************************** * * * Paramètres : gclass = classe GLib à initialiser. * * pyclass = classe Python à initialiser. * * * * Description : Initialise la classe générique des instructions. * * * * Retour : 0 pour indiquer un succès de l'opération. * * * * Remarques : - * * * ******************************************************************************/ static int py_arch_instruction_init_gclass(GArchInstructionClass *gclass, PyTypeObject *pyclass) { PY_CLASS_SET_WRAPPER(gclass->get_encoding, py_arch_instruction_get_encoding_wrapper); PY_CLASS_SET_WRAPPER(gclass->get_keyword, py_arch_instruction_get_keyword_wrapper); return 0; } /****************************************************************************** * * * Paramètres : self = objet à initialiser (théoriquement). * * args = arguments fournis à l'appel. * * kwds = arguments de type key=val fournis. * * * * Description : Initialise une instance sur la base du dérivé de GObject. * * * * Retour : 0. * * * * Remarques : - * * * ******************************************************************************/ static int py_arch_instruction_init(PyObject *self, PyObject *args, PyObject *kwds) { unsigned short int tid; /* Indentifiant unique de type */ int ret; /* Bilan de lecture des args. */ GArchInstruction *instr; /* Instruction à manipuler */ #define ARCH_INSTRUCTION_DOC \ "The ArchInstruction object provides a base class for instructions" \ " of any architecture.\n" \ " operands of any kind for new architectures.\n" \ "\n" \ "Calls to the *__init__* constructor of this abstract object expect"\ " one argument: an unique identifier, as an integer value.\n" \ "\n" \ "The following methods have to be defined for new classes:\n" \ "* pychrysalide.arch.ArchRegister._get_encoding();\n" \ "* pychrysalide.arch.ArchRegister._get_keyword().\n" /* Récupération des paramètres */ ret = PyArg_ParseTuple(args, "H", &tid); if (!ret) return -1; /* Initialisation d'un objet GLib */ ret = forward_pygobjet_init(self); if (ret == -1) return -1; /* Eléments de base */ instr = G_ARCH_INSTRUCTION(pygobject_get(self)); if (!g_arch_instruction_create(instr, tid)) return -1; return 0; } /****************************************************************************** * * * Paramètres : instr = instruction quelconque à consulter. * * * * Description : Indique l'encodage d'une instruction de façon détaillée. * * * * Retour : Description humaine de l'encodage utilisé. * * * * Remarques : - * * * ******************************************************************************/ static char *py_arch_instruction_get_encoding_wrapper(const GArchInstruction *instr) { char *result; /* Encodage à retourner */ PyGILState_STATE gstate; /* Sauvegarde d'environnement */ PyObject *pyobj; /* Objet Python concerné */ PyObject *pyret; /* Bilan d'exécution */ #define PLUGIN_MODULE_GET_ENCODING_WRAPPER PYTHON_WRAPPER_DEF \ ( \ _get_encoding, "$self, /", \ METH_NOARGS, \ "Abstract method describing the encoding related to an" \ " instruction.\n" \ "\n" \ "The result should be the string value.\n" \ ) result = NULL; gstate = PyGILState_Ensure(); pyobj = pygobject_new(G_OBJECT(instr)); if (has_python_method(pyobj, "_get_encoding")) { pyret = run_python_method(pyobj, "_get_encoding", NULL); if (pyret != NULL) { if (!PyUnicode_Check(pyret)) log_variadic_message(LMT_ERROR, _("The returned raw name must be a string")); else result = strdup(PyUnicode_DATA(pyret)); } Py_XDECREF(pyret); } Py_DECREF(pyobj); PyGILState_Release(gstate); return result; } /****************************************************************************** * * * Paramètres : instr = instruction d'assemblage à consulter. * * * * Description : Fournit le nom humain de l'instruction manipulée. * * * * Retour : Mot clef de bas niveau. * * * * Remarques : - * * * ******************************************************************************/ static char *py_arch_instruction_get_keyword_wrapper(const GArchInstruction *instr) { char *result; /* Etiquette à retourner */ PyGILState_STATE gstate; /* Sauvegarde d'environnement */ PyObject *pyobj; /* Objet Python concerné */ PyObject *pyret; /* Bilan d'exécution */ #define PLUGIN_MODULE_GET_KEYWORD_WRAPPER PYTHON_WRAPPER_DEF \ ( \ _get_keyword, "$self, /", \ METH_NOARGS, \ "Abstract method giving the official name of the assembly" \ " instruction.\n" \ "\n" \ "The result should be the string value.\n" \ ) result = NULL; gstate = PyGILState_Ensure(); pyobj = pygobject_new(G_OBJECT(instr)); if (has_python_method(pyobj, "_get_keyword")) { pyret = run_python_method(pyobj, "_get_keyword", NULL); if (pyret != NULL) { if (!PyUnicode_Check(pyret)) log_variadic_message(LMT_ERROR, _("The returned raw name must be a string")); else result = strdup(PyUnicode_DATA(pyret)); } Py_XDECREF(pyret); } Py_DECREF(pyobj); PyGILState_Release(gstate); return result; } /* ---------------------------------------------------------------------------------- */ /* DEFINITION DES LIAISONS ENTRE INSTRUCTIONS */ /* ---------------------------------------------------------------------------------- */ /****************************************************************************** * * * Paramètres : self = instruction d'architecture à manipuler. * * unused = adresse non utilisée ici. * * * * Description : Fournit les origines d'une instruction donnée. * * * * Retour : Nombre de ces origines. * * * * Remarques : - * * * ******************************************************************************/ static PyObject *py_arch_instruction_get_sources(PyObject *self, void *unused) { PyObject *result; /* Instance à retourner */ GArchInstruction *instr; /* Version native */ size_t count; /* Nombre de liens présents */ size_t i; /* Boucle de parcours */ GArchInstruction *src; /* Instruction en source */ InstructionLinkType src_type; /* Type de lien */ PyObject *linked; /* Source de lien Python */ PyObject *lnk_type; /* Nature du lien en Python */ #ifndef NDEBUG int ret; /* Bilan d'une écriture d'arg. */ #endif #define ARCH_INSTRUCTION_SOURCES_ATTRIB PYTHON_GET_DEF_FULL \ ( \ sources, py_arch_instruction, \ "Provide the instructions list driving to the current instruction.\n" \ "\n" \ "Each item of the resulting tuple is a pair of" \ " pychrysalide.arch.ArchInstruction instance and" \ " pychrysalide.arch.ArchInstruction.InstructionLinkType value." \ ) instr = G_ARCH_INSTRUCTION(pygobject_get(self)); g_thick_object_lock(G_THICK_OBJECT(instr)); count = g_arch_instruction_count_src_links(instr); result = PyTuple_New(count); for (i = 0; i < count; i++) { src = g_arch_instruction_get_linked_source(instr, i, &src_type); linked = pygobject_new(G_OBJECT(src)); lnk_type = cast_with_constants_group_from_type(get_python_arch_instruction_type(), "InstructionLinkType", src_type); #ifndef NDEBUG ret = PyTuple_SetItem(result, i, Py_BuildValue("(OO)", linked, lnk_type)); assert(ret == 0); #else PyTuple_SetItem(result, i, Py_BuildValue("(OO)", linked, lnk_type)); #endif unref_object(src); } g_thick_object_unlock(G_THICK_OBJECT(instr)); return result; } /****************************************************************************** * * * Paramètres : self = instruction d'architecture à manipuler. * * unused = adresse non utilisée ici. * * * * Description : Fournit les destinations d'une instruction donnée. * * * * Retour : Nombre de ces destinations. * * * * Remarques : - * * * ******************************************************************************/ static PyObject *py_arch_instruction_get_destinations(PyObject *self, void *unused) { PyObject *result; /* Instance à retourner */ GArchInstruction *instr; /* Version native */ size_t count; /* Nombre de liens présents */ size_t i; /* Boucle de parcours */ GArchInstruction *dest; /* Instruction en source */ InstructionLinkType dest_type; /* Type de lien */ PyObject *linked; /* Destination de lien Python */ PyObject *lnk_type; /* Nature du lien en Python */ #ifndef NDEBUG int ret; /* Bilan d'une écriture d'arg. */ #endif #define ARCH_INSTRUCTION_DESTINATIONS_ATTRIB PYTHON_GET_DEF_FULL \ ( \ destinations, py_arch_instruction, \ "Provide the instructions list following the current instruction.\n" \ "\n" \ "Each item of the resulting tuple is a pair of" \ " pychrysalide.arch.ArchInstruction instance and" \ " pychrysalide.arch.ArchInstruction.InstructionLinkType value." \ ) instr = G_ARCH_INSTRUCTION(pygobject_get(self)); g_thick_object_lock(G_THICK_OBJECT(instr)); count = g_arch_instruction_count_dest_links(instr); result = PyTuple_New(count); for (i = 0; i < count; i++) { dest = g_arch_instruction_get_linked_destination(instr, i, &dest_type); linked = pygobject_new(G_OBJECT(dest)); lnk_type = cast_with_constants_group_from_type(get_python_arch_instruction_type(), "InstructionLinkType", dest_type); #ifndef NDEBUG ret = PyTuple_SetItem(result, i, Py_BuildValue("(OO)", linked, lnk_type)); assert(ret == 0); #else PyTuple_SetItem(result, i, Py_BuildValue("(OO)", linked, lnk_type)); #endif unref_object(dest); } g_thick_object_unlock(G_THICK_OBJECT(instr)); return result; } /* ---------------------------------------------------------------------------------- */ /* MANIPULATION DES OPERANDES */ /* ---------------------------------------------------------------------------------- */ /****************************************************************************** * * * Paramètres : self = architecture concernée par la procédure. * * args = instruction représentant le point de départ. * * * * Description : Attache un opérande supplémentaire à une instruction. * * * * Retour : None. * * * * Remarques : - * * * ******************************************************************************/ static PyObject *py_arch_instruction_attach_operand(PyObject *self, PyObject *args) { PyObject *result; /* Bilan à retourner */ GArchOperand *op; /* Opérande concerné à ajouter */ int ret; /* Bilan de lecture des args. */ GArchInstruction *instr; /* Instruction manipulée */ #define ARCH_INSTRUCTION_ATTACH_OPERAND_METHOD PYTHON_METHOD_DEF \ ( \ attach_operand, "$self, operand, /", \ METH_VARARGS, py_arch_instruction, \ "Add an extra operand to an instruction.\n" \ "\n" \ "The instruction has to be locked during the instruction.\n" \ "\n" \ "The *operand* argument has to be a pychrysalide.arch.ArchOperand" \ " instance." \ ) ret = PyArg_ParseTuple(args, "O&", convert_to_arch_operand, &op); if (!ret) return NULL; instr = G_ARCH_INSTRUCTION(pygobject_get(self)); g_arch_instruction_attach_operand(instr, op); result = Py_None; Py_INCREF(result); return result; } /****************************************************************************** * * * Paramètres : self = architecture concernée par la procédure. * * args = instruction représentant le point de départ. * * * * Description : Remplace un opérande d'une instruction par un autre. * * * * Retour : Bilan de l'opération. * * * * Remarques : - * * * ******************************************************************************/ static PyObject *py_arch_instruction_replace_operand(PyObject *self, PyObject *args) { PyObject *result; /* Bilan à retourner */ GArchOperand *old; /* Ancien opérande à remplacer */ GArchOperand *new; /* Nouvel opérande à intégrer */ int ret; /* Bilan de lecture des args. */ GArchInstruction *instr; /* Instruction manipulée */ bool status; /* Bilan de l'opération */ #define ARCH_INSTRUCTION_REPLACE_OPERAND_METHOD PYTHON_METHOD_DEF \ ( \ replace_operand, "$self, old, new, /", \ METH_VARARGS, py_arch_instruction, \ "Replace an old instruction operand by a another one.\n" \ "\n" \ "The instruction has to be locked during the instruction.\n" \ "\n" \ "Both the *old* and *new* arguments have to be a" \ " pychrysalide.arch.ArchOperand instance.\n" \ "\n" \ "The status of the operation is returned as a boolean value: *True*"\ " if the operand has been replaced, *False* in case of failure." \ ) ret = PyArg_ParseTuple(args, "O&O&", convert_to_arch_operand, &old, convert_to_arch_operand, &new); if (!ret) return NULL; instr = G_ARCH_INSTRUCTION(pygobject_get(self)); status = g_arch_instruction_replace_operand(instr, old, new); result = status ? Py_True : Py_False; Py_INCREF(result); return result; } /****************************************************************************** * * * Paramètres : self = architecture concernée par la procédure. * * args = instruction représentant le point de départ. * * * * Description : Détache un opérande liée d'une instruction. * * * * Retour : Bilan de l'opération. * * * * Remarques : - * * * ******************************************************************************/ static PyObject *py_arch_instruction_detach_operand(PyObject *self, PyObject *args) { PyObject *result; /* Bilan à retourner */ GArchOperand *target; /* Opérande ciblé par l'action */ int ret; /* Bilan de lecture des args. */ GArchInstruction *instr; /* Instruction manipulée */ bool status; /* Bilan de l'opération */ #define ARCH_INSTRUCTION_DETACH_OPERAND_METHOD PYTHON_METHOD_DEF \ ( \ detach_operand, "$self, operand, /", \ METH_VARARGS, py_arch_instruction, \ "Remove an operand from the instruction.\n" \ "\n" \ "The instruction has to be locked during the instruction.\n" \ "\n" \ "The *operand* argument has to be a pychrysalide.arch.ArchOperand" \ " instance.\n" \ "\n" \ "The status of the operation is returned as a boolean value: *True*"\ " if the operand has been removed, *False* in case of failure." \ ) ret = PyArg_ParseTuple(args, "O&", convert_to_arch_operand, &target); if (!ret) return NULL; instr = G_ARCH_INSTRUCTION(pygobject_get(self)); status = g_arch_instruction_detach_operand(instr, target); result = status ? Py_True : Py_False; Py_INCREF(result); return result; } /****************************************************************************** * * * Paramètres : self = architecture concernée par la procédure. * * args = instruction représentant le point de départ. * * * * Description : Détermine le chemin conduisant à un opérande. * * * * Retour : Chemin d'accès à l'opérande ou None en cas d'absence. * * * * Remarques : - * * * ******************************************************************************/ static PyObject *py_arch_instruction_find_operand_path(PyObject *self, PyObject *args) { PyObject *result; /* Chemin à retourner */ GArchOperand *target; /* Opérande ciblé par l'action */ int ret; /* Bilan de lecture des args. */ GArchInstruction *instr; /* Instruction manipulée */ char *path; /* Chemin déterminé */ #define ARCH_INSTRUCTION_FIND_OPERAND_PATH_METHOD PYTHON_METHOD_DEF \ ( \ find_operand_path, "$self, target, /", \ METH_VARARGS, py_arch_instruction, \ "Compute the path leading to an instruction operand.\n" \ "\n" \ "The *target* has to be an instance of pychrysalide.arch.ArchOperand" \ " included in the instruction.\n" \ "\n" \ "The result is a string of the form 'n[:n:n:n]', where n is an" \ " internal index, or None if the *target* is not found. This kind of" \ " path is aimed to be built for the" \ " pychrysalide.arch.ArchInstruction.find_operand_path() function." \ ) ret = PyArg_ParseTuple(args, "O&", convert_to_arch_operand, &target); if (!ret) return NULL; instr = G_ARCH_INSTRUCTION(pygobject_get(self)); path = g_arch_instruction_find_operand_path(instr, target); if (path != NULL) { result = PyUnicode_FromString(path); free(path); } else { result = Py_None; Py_INCREF(result); } return result; } /****************************************************************************** * * * Paramètres : self = architecture concernée par la procédure. * * args = instruction représentant le point de départ. * * * * Description : Obtient l'opérande correspondant à un chemin donné. * * * * Retour : Opérande trouvé ou None en cas d'échec. * * * * Remarques : - * * * ******************************************************************************/ static PyObject *py_arch_instruction_get_operand_from_path(PyObject *self, PyObject *args) { PyObject *result; /* Trouvaille à retourner */ const char *path; /* Chemin à parcourir */ int ret; /* Bilan de lecture des args. */ GArchInstruction *instr; /* Instruction manipulée */ GArchOperand *op; /* Opérande retrouvé */ #define ARCH_INSTRUCTION_GET_OPERAND_FROM_PATH_METHOD PYTHON_METHOD_DEF \ ( \ get_operand_from_path, "$self, path, /", \ METH_VARARGS, py_arch_instruction, \ "Retrieve an operand from an instruction by its path.\n" \ "\n" \ "This *path* is a string of the form 'n[:n:n:n]', where n is an" \ " internal index. Such a path is usually built by the" \ " pychrysalide.arch.ArchInstruction.find_operand_path() function.\n" \ "\n" \ "The result is an pychrysalide.arch.ArchOperand instance, or" \ " None if no operand was found." \ ) ret = PyArg_ParseTuple(args, "s", &path); if (!ret) return NULL; instr = G_ARCH_INSTRUCTION(pygobject_get(self)); op = g_arch_instruction_get_operand_from_path(instr, path); if (op != NULL) { result = pygobject_new(G_OBJECT(op)); unref_object(op); } else { result = Py_None; Py_INCREF(result); } return result; } /****************************************************************************** * * * Paramètres : self = objet représentant une instruction. * * unused = adresse non utilisée ici. * * * * Description : Fournit tous les opérandes d'une instruction. * * * * Retour : Valeur associée à la propriété consultée. * * * * Remarques : - * * * ******************************************************************************/ static PyObject *py_arch_instruction_get_operands(PyObject *self, void *unused) { PyObject *result; /* Instance à retourner */ GArchInstruction *instr; /* Version native */ size_t count; /* Nombre d'opérandes présents */ size_t i; /* Boucle de parcours */ GArchOperand *operand; /* Opérande à manipuler */ PyObject *opobj; /* Version Python */ #ifndef NDEBUG int ret; /* Bilan d'une écriture d'arg. */ #endif #define ARCH_INSTRUCTION_OPERANDS_ATTRIB PYTHON_GET_DEF_FULL \ ( \ operands, py_arch_instruction, \ "List of instruction attached operands.\n" \ "\n" \ "The result is a list of pychrysalide.arch.ArchOperand" \ " instances, which can be empty." \ ) instr = G_ARCH_INSTRUCTION(pygobject_get(self)); g_thick_object_lock(G_THICK_OBJECT(instr)); count = g_arch_instruction_count_operands(instr); result = PyTuple_New(count); for (i = 0; i < count; i++) { operand = g_arch_instruction_get_operand(instr, i); opobj = pygobject_new(G_OBJECT(operand)); #ifndef NDEBUG ret = PyTuple_SetItem(result, i, Py_BuildValue("O", opobj)); assert(ret == 0); #else PyTuple_SetItem(result, i, Py_BuildValue("O", opobj)); #endif unref_object(operand); } g_thick_object_unlock(G_THICK_OBJECT(instr)); return result; } /* ---------------------------------------------------------------------------------- */ /* LIAISON DE FONCTIONNALITES AVEC L'API PYTHON */ /* ---------------------------------------------------------------------------------- */ /****************************************************************************** * * * Paramètres : self = architecture concernée par la procédure. * * args = instruction représentant le point de départ. * * * * Description : Ajoute une information complémentaire à une instruction. * * * * Retour : Bilan de l'opération. * * * * Remarques : - * * * ******************************************************************************/ static PyObject *py_arch_instruction_set_flag(PyObject *self, PyObject *args) { PyObject *result; /* Bilan à retourner */ unsigned int flag; /* Fanion(s) à appliquer */ int ret; /* Bilan de lecture des args. */ GArchInstruction *instr; /* Instruction manipulée */ bool status; /* Bilan à transmettre */ #define ARCH_INSTRUCTION_SET_FLAG_METHOD PYTHON_METHOD_DEF \ ( \ set_flag, "$self, flag, /", \ METH_VARARGS, py_arch_instruction, \ "Add some flags to the instruction.\n" \ "\n" \ "This *flag* argument is an integer value containing" \ " bits to apply to the instruction state.\n" \ "\n" \ "The result is an boolean status: *True* for operation" \ " success, *False* otherwise." \ ) ret = PyArg_ParseTuple(args, "I", &flag); if (!ret) return NULL; instr = G_ARCH_INSTRUCTION(pygobject_get(self)); status = g_arch_instruction_set_flag(instr, flag); result = status ? Py_True : Py_False; Py_INCREF(result); return result; } /****************************************************************************** * * * Paramètres : self = architecture concernée par la procédure. * * args = instruction représentant le point de départ. * * * * Description : Retire une information complémentaire à une instruction. * * * * Retour : Bilan de l'opération. * * * * Remarques : - * * * ******************************************************************************/ static PyObject *py_arch_instruction_unset_flag(PyObject *self, PyObject *args) { PyObject *result; /* Bilan à retourner */ unsigned int flag; /* Fanion(s) à appliquer */ int ret; /* Bilan de lecture des args. */ GArchInstruction *instr; /* Instruction manipulée */ bool status; /* Bilan à transmettre */ #define ARCH_INSTRUCTION_UNSET_FLAG_METHOD PYTHON_METHOD_DEF \ ( \ unset_flag, "$self, flag, /", \ METH_VARARGS, py_arch_instruction, \ "Remove some flags from the instruction.\n" \ "\n" \ "This *flag* argument is an integer value containing" \ " bits to delete from the instruction state.\n" \ "\n" \ "The result is an boolean status: *True* for operation" \ " success, *False* otherwise." \ ) ret = PyArg_ParseTuple(args, "I", &flag); if (!ret) return NULL; instr = G_ARCH_INSTRUCTION(pygobject_get(self)); status = g_arch_instruction_unset_flag(instr, flag); result = status ? Py_True : Py_False; Py_INCREF(result); return result; } /****************************************************************************** * * * Paramètres : self = architecture concernée par la procédure. * * args = instruction représentant le point de départ. * * * * Description : Détermine si une instruction possède un fanion particulier. * * * * Retour : Bilan de l'opération. * * * * Remarques : - * * * ******************************************************************************/ static PyObject *py_arch_instruction_has_flag(PyObject *self, PyObject *args) { PyObject *result; /* Bilan à retourner */ unsigned int flag; /* Fanion(s) à appliquer */ int ret; /* Bilan de lecture des args. */ GArchInstruction *instr; /* Instruction manipulée */ bool status; /* Bilan à transmettre */ #define ARCH_INSTRUCTION_HAS_FLAG_METHOD PYTHON_METHOD_DEF \ ( \ has_flag, "$self, flag, /", \ METH_VARARGS, py_arch_instruction, \ "Tell if some flags are set for the instruction.\n" \ "\n" \ "This *flag* argument is an integer value containing" \ " bits to test for the instruction state.\n" \ "\n" \ "The result is an boolean status: *True* if the bits" \ " are active, *False* otherwise." \ ) ret = PyArg_ParseTuple(args, "I", &flag); if (!ret) return NULL; instr = G_ARCH_INSTRUCTION(pygobject_get(self)); status = g_arch_instruction_has_flag(instr, flag); result = status ? Py_True : Py_False; Py_INCREF(result); return result; } /****************************************************************************** * * * Paramètres : self = objet Python concerné par l'appel. * * closure = non utilisé ici. * * * * Description : Fournit l'identifiant correspondant à un type d'instructions.* * * * Retour : Identifiant unique par type d'instruction et architecture. * * * * Remarques : - * * * ******************************************************************************/ static PyObject *py_arch_instruction_get_type_id(PyObject *self, void *closure) { PyObject *result; /* Valeur à retourner */ GArchInstruction *instr; /* Instruction manipulée */ itid_t tid; /* Identifiant à transmettre */ #define ARCH_INSTRUCTION_TYPE_ID_ATTRIB PYTHON_GET_DEF_FULL \ ( \ type_id, py_arch_instruction, \ "Provide the unique identifier given to this kind of" \ " instruction.\n" \ "\n" \ "The returned value is an integer." \ ) instr = G_ARCH_INSTRUCTION(pygobject_get(self)); tid = g_arch_instruction_get_type_id(instr); result = PyLong_FromUnsignedLong(tid); return result; } /****************************************************************************** * * * Paramètres : self = objet Python concerné par l'appel. * * closure = non utilisé ici. * * * * Description : Indique l'encodage d'une instruction de façon détaillée. * * * * Retour : Description humaine de l'encodage utilisé. * * * * Remarques : - * * * ******************************************************************************/ static PyObject *py_arch_instruction_get_encoding(PyObject *self, void *closure) { PyObject *result; /* Valeur à retourner */ GArchInstruction *instr; /* Instruction manipulée */ char *encoding; /* Encodage d'une instruction */ #define ARCH_INSTRUCTION_ENCODING_ATTRIB PYTHON_GET_DEF_FULL \ ( \ encoding, py_arch_instruction, \ "Describe the encoding related to an instruction.\n" \ "\n" \ "The returned value is an arbitrary string value." \ ) instr = G_ARCH_INSTRUCTION(pygobject_get(self)); encoding = g_arch_instruction_get_encoding(instr); if (encoding != NULL) { result = PyUnicode_FromString(encoding); free(encoding); } else { /** * La méthode de classe sollicitée a renvoyé une valeur nulle. * * Si cette méthode correspond à une implémentation Python * (avec un appel à not_yet_implemented_method()), une exception * est déjà en place. * * Si aucune exception n'a été prévue, un rattrapage est effectué ici. */ if (PyErr_Occurred() == NULL) PyErr_SetString(PyExc_NotImplementedError, _("unexpected NULL value as encoding")); result = NULL; } return result; } /****************************************************************************** * * * Paramètres : self = objet Python concerné par l'appel. * * closure = non utilisé ici. * * * * Description : Indique l'encodage d'une instruction de façon détaillée. * * * * Retour : Description humaine de l'encodage utilisé. * * * * Remarques : - * * * ******************************************************************************/ static PyObject *py_arch_instruction_get_keyword(PyObject *self, void *closure) { PyObject *result; /* Valeur à retourner */ GArchInstruction *instr; /* Instruction manipulée */ char *keyword; /* Désignation d'une instruct° */ #define ARCH_INSTRUCTION_KEYWORD_ATTRIB PYTHON_GET_DEF_FULL \ ( \ keyword, py_arch_instruction, \ "Give the official name of the assembly instruction.\n" \ "\n" \ "The returned value is a string value." \ ) instr = G_ARCH_INSTRUCTION(pygobject_get(self)); keyword = g_arch_instruction_get_keyword(instr); if (keyword != NULL) { result = PyUnicode_FromString(keyword); free(keyword); } else { /** * La méthode de classe sollicitée a renvoyé une valeur nulle. * * Si cette méthode correspond à une implémentation Python * (avec un appel à not_yet_implemented_method()), une exception * est déjà en place. * * Si aucune exception n'a été prévue, un rattrapage est effectué ici. */ if (PyErr_Occurred() == NULL) PyErr_SetString(PyExc_NotImplementedError, _("unexpected NULL value as keyword")); result = NULL; } return result; } /****************************************************************************** * * * Paramètres : self = objet Python concerné par l'appel. * * closure = non utilisé ici. * * * * Description : Fournit la place mémoire d'une instruction. * * * * Retour : Définition de localisation ou *None*. * * * * Remarques : - * * * ******************************************************************************/ static PyObject *py_arch_instruction_get_range(PyObject *self, void *closure) { PyObject *result; /* Valeur à retourner */ GArchInstruction *instr; /* Instruction manipulée */ mrange_t range; /* Localisation d'instruction */ bool valid; /* Validité de la localisation */ #define ARCH_INSTRUCTION_RANGE_ATTRIB PYTHON_GET_DEF_FULL \ ( \ range, py_arch_instruction, \ "Give access to the memory range covered by the" \ " current instruction.\n" \ "\n" \ "The returned value is a pychrysalide.arch.mrange" \ " instance or *None* if no location is currently" \ " defined." \ ) instr = G_ARCH_INSTRUCTION(pygobject_get(self)); valid = g_arch_instruction_get_range(instr, &range); if (valid) result = build_from_internal_mrange(&range); else { result = Py_None; Py_INCREF(result); } return result; } /****************************************************************************** * * * Paramètres : - * * * * Description : Fournit un accès à une définition de type à diffuser. * * * * Retour : Définition d'objet pour Python. * * * * Remarques : - * * * ******************************************************************************/ PyTypeObject *get_python_arch_instruction_type(void) { static PyMethodDef py_arch_instruction_methods[] = { ARCH_INSTRUCTION_ATTACH_OPERAND_METHOD, ARCH_INSTRUCTION_REPLACE_OPERAND_METHOD, ARCH_INSTRUCTION_DETACH_OPERAND_METHOD, ARCH_INSTRUCTION_FIND_OPERAND_PATH_METHOD, ARCH_INSTRUCTION_GET_OPERAND_FROM_PATH_METHOD, ARCH_INSTRUCTION_SET_FLAG_METHOD, ARCH_INSTRUCTION_UNSET_FLAG_METHOD, ARCH_INSTRUCTION_HAS_FLAG_METHOD, { NULL } }; static PyGetSetDef py_arch_instruction_getseters[] = { ARCH_INSTRUCTION_SOURCES_ATTRIB, ARCH_INSTRUCTION_DESTINATIONS_ATTRIB, ARCH_INSTRUCTION_OPERANDS_ATTRIB, ARCH_INSTRUCTION_TYPE_ID_ATTRIB, ARCH_INSTRUCTION_ENCODING_ATTRIB, ARCH_INSTRUCTION_KEYWORD_ATTRIB, ARCH_INSTRUCTION_RANGE_ATTRIB, { NULL } }; static PyTypeObject py_arch_instruction_type = { PyVarObject_HEAD_INIT(NULL, 0) .tp_name = "pychrysalide.arch.ArchInstruction", .tp_basicsize = sizeof(PyGObject), .tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_IS_ABSTRACT | Py_TPFLAGS_BASETYPE, .tp_doc = ARCH_INSTRUCTION_DOC, .tp_methods = py_arch_instruction_methods, .tp_getset = py_arch_instruction_getseters, .tp_init = py_arch_instruction_init, .tp_new = py_arch_instruction_new, }; return &py_arch_instruction_type; } /****************************************************************************** * * * Paramètres : module = module dont la définition est à compléter. * * * * Description : Prend en charge l'objet 'pychrysalide.arch.ArchInstruction'. * * * * Retour : Bilan de l'opération. * * * * Remarques : - * * * ******************************************************************************/ bool ensure_python_arch_instruction_is_registered(void) { PyTypeObject *type; /* Type Python 'ArchInstruc...'*/ PyObject *module; /* Module à recompléter */ PyObject *dict; /* Dictionnaire du module */ type = get_python_arch_instruction_type(); if (!PyType_HasFeature(type, Py_TPFLAGS_READY)) { module = get_access_to_python_module("pychrysalide.arch"); dict = PyModule_GetDict(module); if (!ensure_python_thick_object_is_registered()) return false; if (!ensure_python_serializable_object_is_registered()) return false; pyg_register_class_init(G_TYPE_ARCH_INSTRUCTION, (PyGClassInitFunc)py_arch_instruction_init_gclass); if (!register_class_for_pygobject(dict, G_TYPE_ARCH_INSTRUCTION, type)) return false; if (!define_arch_instruction_constants(type)) return false; } return true; } /****************************************************************************** * * * 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 instruction d'architecture. * * * * Retour : Bilan de l'opération, voire indications supplémentaires. * * * * Remarques : - * * * ******************************************************************************/ int convert_to_arch_instruction(PyObject *arg, void *dst) { int result; /* Bilan à retourner */ result = PyObject_IsInstance(arg, (PyObject *)get_python_arch_instruction_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 arch instruction"); break; case 1: *((GArchInstruction **)dst) = G_ARCH_INSTRUCTION(pygobject_get(arg)); break; default: assert(false); break; } return result; }