summaryrefslogtreecommitdiff
path: root/plugins
diff options
context:
space:
mode:
authorCyrille Bagard <nocbos@gmail.com>2019-01-26 09:19:53 (GMT)
committerCyrille Bagard <nocbos@gmail.com>2019-01-26 09:19:53 (GMT)
commitdff8fd4c83d9833d4539e2dd0c9d3f998b73608f (patch)
tree5e1088e0572b1e921df6d31d3db3ef753d97555b /plugins
parentae3725eae3c1f2008e6f26b1057d97a6b9050a3c (diff)
Extended the Python API.
Diffstat (limited to 'plugins')
-rw-r--r--plugins/pychrysalide/arch/instruction.c265
-rw-r--r--plugins/pychrysalide/arch/operand.c45
-rw-r--r--plugins/pychrysalide/arch/operand.h3
-rw-r--r--plugins/pychrysalide/glibext/Makefile.am1
-rw-r--r--plugins/pychrysalide/glibext/buffercache.c4
-rw-r--r--plugins/pychrysalide/glibext/buffercache.h4
-rw-r--r--plugins/pychrysalide/glibext/bufferview.c156
-rw-r--r--plugins/pychrysalide/glibext/bufferview.h42
-rw-r--r--plugins/pychrysalide/glibext/module.c2
9 files changed, 458 insertions, 64 deletions
diff --git a/plugins/pychrysalide/arch/instruction.c b/plugins/pychrysalide/arch/instruction.c
index 881c6ff..fb2df5b 100644
--- a/plugins/pychrysalide/arch/instruction.c
+++ b/plugins/pychrysalide/arch/instruction.c
@@ -35,6 +35,7 @@
#include <plugins/dt.h>
+#include "operand.h"
#include "vmpa.h"
#include "../access.h"
#include "../helpers.h"
@@ -105,6 +106,23 @@ static int py_arch_instruction_init(PyObject *, PyObject *, PyObject *);
+/* --------------------------- MANIPULATION DES OPERANDES --------------------------- */
+
+
+/* Attache un opérande supplémentaire à une instruction. */
+static PyObject *py_arch_instruction_attach_extra_operand(PyObject *, PyObject *);
+
+/* Fournit tous les opérandes d'une instruction. */
+static PyObject *py_arch_instruction_get_operands(PyObject *, void *);
+
+/* 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 *);
+
+
+
/* ------------------- DEFINITION DES LIAISONS ENTRE INSTRUCTIONS ------------------- */
@@ -131,9 +149,6 @@ static int py_arch_instruction_set_range(PyObject *, PyObject *, void *);
/* Fournit le nom humain de l'instruction manipulée. */
static PyObject *py_arch_instruction_get_keyword(PyObject *, void *);
-/* Fournit tous les opérandes d'une instruction. */
-static PyObject *py_arch_instruction_get_operands(PyObject *, void *);
-
/* Définit les constantes pour les instructions. */
static bool py_arch_instruction_define_constants(PyTypeObject *);
@@ -401,6 +416,178 @@ static int py_arch_instruction_init(PyObject *self, PyObject *args, PyObject *kw
/* ---------------------------------------------------------------------------------- */
+/* 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_extra_operand(PyObject *self, PyObject *args)
+{
+ GArchOperand *op; /* Opérande concerné à ajouter */
+ int ret; /* Bilan de lecture des args. */
+ GArchInstruction *instr; /* Instruction manipulée */
+
+ ret = PyArg_ParseTuple(args, "O&", convert_to_arch_operand, &op);
+ if (!ret) return NULL;
+
+ instr = G_ARCH_INSTRUCTION(pygobject_get(self));
+
+ g_object_ref(G_OBJECT(op));
+
+ g_arch_instruction_attach_extra_operand(instr, op);
+
+ Py_RETURN_NONE;
+
+}
+
+
+/******************************************************************************
+* *
+* 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
+
+ instr = G_ARCH_INSTRUCTION(pygobject_get(self));
+
+ g_arch_instruction_lock_operands(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
+
+ g_object_unref(G_OBJECT(operand));
+
+ }
+
+ g_arch_instruction_unlock_operands(instr);
+
+ 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 */
+
+ 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);
+
+ if (status)
+ g_object_ref(G_OBJECT(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 */
+
+ 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;
+
+}
+
+
+
+/* ---------------------------------------------------------------------------------- */
/* DEFINITION DES LIAISONS ENTRE INSTRUCTIONS */
/* ---------------------------------------------------------------------------------- */
@@ -652,63 +839,6 @@ static PyObject *py_arch_instruction_get_keyword(PyObject *self, void *unused)
/******************************************************************************
* *
-* 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
-
- instr = G_ARCH_INSTRUCTION(pygobject_get(self));
-
- g_arch_instruction_lock_operands(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
-
- g_object_unref(G_OBJECT(operand));
-
- }
-
- g_arch_instruction_unlock_operands(instr);
-
- return result;
-
-}
-
-
-/******************************************************************************
-* *
* Paramètres : obj_type = type dont le dictionnaire est à compléter. *
* *
* Description : Définit les constantes pour les instructions. *
@@ -755,6 +885,21 @@ static bool py_arch_instruction_define_constants(PyTypeObject *obj_type)
PyTypeObject *get_python_arch_instruction_type(void)
{
static PyMethodDef py_arch_instruction_methods[] = {
+ {
+ "attach_operand", py_arch_instruction_attach_extra_operand,
+ METH_VARARGS,
+ "attach_operand($self, op, /)\n--\n\nAdd a new operand to the instruction."
+ },
+ {
+ "replace_operand", py_arch_instruction_replace_operand,
+ METH_VARARGS,
+ "replace_operand($self, old, new, /)\n--\n\nReplace an old instruction operand by a another one."
+ },
+ {
+ "detach_operand", py_arch_instruction_detach_operand,
+ METH_VARARGS,
+ "detach_operand($self, target, /)\n--\n\nRemove an operand from the instruction."
+ },
{ NULL }
};
diff --git a/plugins/pychrysalide/arch/operand.c b/plugins/pychrysalide/arch/operand.c
index 4630b3a..6780e10 100644
--- a/plugins/pychrysalide/arch/operand.c
+++ b/plugins/pychrysalide/arch/operand.c
@@ -223,3 +223,48 @@ bool ensure_python_arch_operand_is_registered(void)
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 opérande d'architecture. *
+* *
+* Retour : Bilan de l'opération, voire indications supplémentaires. *
+* *
+* Remarques : - *
+* *
+******************************************************************************/
+
+int convert_to_arch_operand(PyObject *arg, void *dst)
+{
+ int result; /* Bilan à retourner */
+
+ result = PyObject_IsInstance(arg, (PyObject *)get_python_arch_operand_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 operand");
+ break;
+
+ case 1:
+ *((GArchOperand **)dst) = G_ARCH_OPERAND(pygobject_get(arg));
+ break;
+
+ default:
+ assert(false);
+ break;
+
+ }
+
+ return result;
+
+}
diff --git a/plugins/pychrysalide/arch/operand.h b/plugins/pychrysalide/arch/operand.h
index a718d8f..595a4fc 100644
--- a/plugins/pychrysalide/arch/operand.h
+++ b/plugins/pychrysalide/arch/operand.h
@@ -37,6 +37,9 @@ PyTypeObject *get_python_arch_operand_type(void);
/* Prend en charge l'objet 'pychrysalide.arch.ArchOperand'. */
bool ensure_python_arch_operand_is_registered(void);
+/* Tente de convertir en opérande d'architecture. */
+int convert_to_arch_operand(PyObject *, void *);
+
#endif /* _PLUGINS_PYCHRYSALIDE_ARCH_OPERAND_H */
diff --git a/plugins/pychrysalide/glibext/Makefile.am b/plugins/pychrysalide/glibext/Makefile.am
index c09f4fa..086ca28 100644
--- a/plugins/pychrysalide/glibext/Makefile.am
+++ b/plugins/pychrysalide/glibext/Makefile.am
@@ -4,6 +4,7 @@ noinst_LTLIBRARIES = libpychrysaglibext.la
libpychrysaglibext_la_SOURCES = \
buffercache.h buffercache.c \
bufferline.h bufferline.c \
+ bufferview.h bufferview.c \
configuration.h configuration.c \
linecursor.h linecursor.c \
linegen.h linegen.c \
diff --git a/plugins/pychrysalide/glibext/buffercache.c b/plugins/pychrysalide/glibext/buffercache.c
index 8818f5e..6f07f4e 100644
--- a/plugins/pychrysalide/glibext/buffercache.c
+++ b/plugins/pychrysalide/glibext/buffercache.c
@@ -1,6 +1,6 @@
/* Chrysalide - Outil d'analyse de fichiers binaires
- * buffercache.c - équivalent Python du fichier "glibext/gbuffercache.h"
+ * buffercache.c - équivalent Python du fichier "glibext/gbuffercache.c"
*
* Copyright (C) 2016-2017 Cyrille Bagard
*
@@ -120,7 +120,7 @@ PyTypeObject *get_python_buffer_cache_type(void)
PyVarObject_HEAD_INIT(NULL, 0)
- .tp_name = "pychrysalide.glibext.Buffercache",
+ .tp_name = "pychrysalide.glibext.BufferCache",
.tp_basicsize = sizeof(PyGObject),
.tp_flags = Py_TPFLAGS_DEFAULT,
diff --git a/plugins/pychrysalide/glibext/buffercache.h b/plugins/pychrysalide/glibext/buffercache.h
index 728acd7..c0e4493 100644
--- a/plugins/pychrysalide/glibext/buffercache.h
+++ b/plugins/pychrysalide/glibext/buffercache.h
@@ -1,6 +1,6 @@
/* Chrysalide - Outil d'analyse de fichiers binaires
- * buffercache.h - prototypes pour l'équivalent Python du fichier "glibext/buffercache.h"
+ * buffercache.h - prototypes pour l'équivalent Python du fichier "glibext/gbuffercache.h"
*
* Copyright (C) 2016-2017 Cyrille Bagard
*
@@ -34,7 +34,7 @@
/* Fournit un accès à une définition de type à diffuser. */
PyTypeObject *get_python_buffer_cache_type(void);
-/* Prend en charge l'objet 'pychrysalide.glibext.CodeBuffer'. */
+/* Prend en charge l'objet 'pychrysalide.glibext.BufferCache'. */
bool ensure_python_buffer_cache_is_registered(void);
diff --git a/plugins/pychrysalide/glibext/bufferview.c b/plugins/pychrysalide/glibext/bufferview.c
new file mode 100644
index 0000000..1c8d7cc
--- /dev/null
+++ b/plugins/pychrysalide/glibext/bufferview.c
@@ -0,0 +1,156 @@
+
+/* Chrysalide - Outil d'analyse de fichiers binaires
+ * bufferview.c - équivalent Python du fichier "glibext/gbufferview.c"
+ *
+ * Copyright (C) 2018 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 "bufferview.h"
+
+
+#include <pygobject.h>
+
+
+#include <glibext/gbufferview.h>
+
+
+#include "../access.h"
+#include "../helpers.h"
+
+
+
+/* Fournit le tampon de code lié à un visualisateur donné. */
+static PyObject *py_buffer_view_get_cache(PyObject *, void *);
+
+
+
+/******************************************************************************
+* *
+* Paramètres : self = objet Python concerné par l'appel. *
+* closure = non utilisé ici. *
+* *
+* Description : Fournit le tampon de code lié à un visualisateur donné. *
+* *
+* Retour : Tampon de code associé au gestionnaire d'affichage. *
+* *
+* Remarques : - *
+* *
+******************************************************************************/
+
+static PyObject *py_buffer_view_get_cache(PyObject *self, void *closure)
+{
+ PyObject *result; /* Instance Python à retourner */
+ GBufferView *view; /* Vue GLib à consulter */
+ GBufferCache *cache; /* Tampon associé à cette vue */
+
+ view = G_BUFFER_VIEW(pygobject_get(self));
+
+ cache = g_buffer_view_get_cache(view);
+
+ result = pygobject_new(G_OBJECT(cache));
+
+ g_object_unref(G_OBJECT(cache));
+
+ 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_buffer_view_type(void)
+{
+ static PyMethodDef py_buffer_view_methods[] = {
+ { NULL }
+ };
+
+ static PyGetSetDef py_buffer_view_getseters[] = {
+ {
+ "cache", py_buffer_view_get_cache, NULL,
+ "Provide the buffer cache for the view.", NULL
+ },
+ { NULL }
+ };
+
+ static PyTypeObject py_buffer_view_type = {
+
+ PyVarObject_HEAD_INIT(NULL, 0)
+
+ .tp_name = "pychrysalide.glibext.BufferView",
+ .tp_basicsize = sizeof(PyGObject),
+
+ .tp_flags = Py_TPFLAGS_DEFAULT,
+
+ .tp_doc = "PyChrysalide code buffer",
+
+ .tp_methods = py_buffer_view_methods,
+ .tp_getset = py_buffer_view_getseters,
+
+ };
+
+ return &py_buffer_view_type;
+
+}
+
+
+/******************************************************************************
+* *
+* Paramètres : module = module dont la définition est à compléter. *
+* *
+* Description : Prend en charge l'objet 'pychrysalide.glibext.BufferView'. *
+* *
+* Retour : Bilan de l'opération. *
+* *
+* Remarques : - *
+* *
+******************************************************************************/
+
+bool ensure_python_buffer_view_is_registered(void)
+{
+ PyTypeObject *type; /* Type Python 'BufferView' */
+ PyObject *module; /* Module à recompléter */
+ PyObject *dict; /* Dictionnaire du module */
+
+ type = get_python_buffer_view_type();
+
+ if (!PyType_HasFeature(type, Py_TPFLAGS_READY))
+ {
+ module = get_access_to_python_module("pychrysalide.glibext");
+
+ dict = PyModule_GetDict(module);
+
+ if (!register_class_for_pygobject(dict, G_TYPE_BUFFER_VIEW, type, &PyGObject_Type))
+ return false;
+
+ }
+
+ return true;
+
+}
diff --git a/plugins/pychrysalide/glibext/bufferview.h b/plugins/pychrysalide/glibext/bufferview.h
new file mode 100644
index 0000000..2ac08b9
--- /dev/null
+++ b/plugins/pychrysalide/glibext/bufferview.h
@@ -0,0 +1,42 @@
+
+/* Chrysalide - Outil d'analyse de fichiers binaires
+ * bufferview.h - prototypes pour l'équivalent Python du fichier "glibext/gbufferview.h"
+ *
+ * Copyright (C) 2018 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
+ */
+
+
+#ifndef _PLUGINS_PYCHRYSALIDE_GLIBEXT_BUFFERVIEW_H
+#define _PLUGINS_PYCHRYSALIDE_GLIBEXT_BUFFERVIEW_H
+
+
+#include <Python.h>
+#include <stdbool.h>
+
+
+
+/* Fournit un accès à une définition de type à diffuser. */
+PyTypeObject *get_python_buffer_view_type(void);
+
+/* Prend en charge l'objet 'pychrysalide.glibext.BufferView'. */
+bool ensure_python_buffer_view_is_registered(void);
+
+
+
+#endif /* _PLUGINS_PYCHRYSALIDE_GLIBEXT_BUFFERVIEW_H */
diff --git a/plugins/pychrysalide/glibext/module.c b/plugins/pychrysalide/glibext/module.c
index 2429dbd..f1c0bbb 100644
--- a/plugins/pychrysalide/glibext/module.c
+++ b/plugins/pychrysalide/glibext/module.c
@@ -30,6 +30,7 @@
#include "buffercache.h"
#include "bufferline.h"
+#include "bufferview.h"
#include "configuration.h"
#include "linecursor.h"
#include "linegen.h"
@@ -95,6 +96,7 @@ bool populate_glibext_module(void)
if (result) result = ensure_python_buffer_cache_is_registered();
if (result) result = ensure_python_buffer_line_is_registered();
+ if (result) result = ensure_python_buffer_view_is_registered();
if (result) result = ensure_python_config_param_is_registered();
if (result) result = ensure_python_config_param_iterator_is_registered();
if (result) result = ensure_python_generic_config_is_registered();