mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2025-01-06 12:09:26 +08:00
b1ce65684d
When the gdbpy_ref objects get destroyed, they call Py_DECREF to decrement the reference counter of the python object they hold a reference to. Any time we call into the Python API, we should be holding the GIL. The gdbpy_enter object does that for us in an RAII-fashion. However, if gdbpy_enter is declared after a gdbpy_ref object in a function, gdbpy_enter's destructor will be called (and the GIL will be released) before gdbpy_ref's destructor is called. Therefore, we will end up calling Py_DECREF without holding the GIL. This became obvious with Python 3.6, where memory management functions have asserts to make sure that the GIL is held. This was exposed by tests py-as-string.exp, py-function.exp and py-xmethods. For example: (gdb) p $_as_string(enum_valid) Fatal Python error: Python memory allocator called without holding the GIL Current thread 0x00007f7f7b21c780 (most recent call first): [1] 18678 abort (core dumped) ./gdb -nx testsuite/outputs/gdb.python/py-as-string/py-as-string #0 0x00007ffff618bc37 in raise () from /lib/x86_64-linux-gnu/libc.so.6 #1 0x00007ffff618f028 in abort () from /lib/x86_64-linux-gnu/libc.so.6 #2 0x00007ffff6b104d6 in Py_FatalError (msg=msg@entry=0x7ffff6ba15b8 "Python memory allocator called without holding the GIL") at Python/pylifecycle.c:1457 #3 0x00007ffff6a37a68 in _PyMem_DebugCheckGIL () at Objects/obmalloc.c:1972 #4 0x00007ffff6a3804e in _PyMem_DebugFree (ctx=0x7ffff6e65290 <_PyMem_Debug+48>, ptr=0x24f8830) at Objects/obmalloc.c:1994 #5 0x00007ffff6a38e1d in PyMem_Free (ptr=<optimized out>) at Objects/obmalloc.c:442 #6 0x00007ffff6b866c6 in _PyFaulthandler_Fini () at ./Modules/faulthandler.c:1369 #7 0x00007ffff6b104bd in Py_FatalError (msg=msg@entry=0x7ffff6ba15b8 "Python memory allocator called without holding the GIL") at Python/pylifecycle.c:1431 #8 0x00007ffff6a37a68 in _PyMem_DebugCheckGIL () at Objects/obmalloc.c:1972 #9 0x00007ffff6a3804e in _PyMem_DebugFree (ctx=0x7ffff6e652c0 <_PyMem_Debug+96>, ptr=0x7ffff46b6040) at Objects/obmalloc.c:1994 #10 0x00007ffff6a38f55 in PyObject_Free (ptr=<optimized out>) at Objects/obmalloc.c:503 #11 0x00007ffff6a5f27e in unicode_dealloc (unicode=unicode@entry=0x7ffff46b6040) at Objects/unicodeobject.c:1794 #12 0x00007ffff6a352a9 in _Py_Dealloc (op=0x7ffff46b6040) at Objects/object.c:1786 #13 0x000000000063f28b in gdb_Py_DECREF (op=0x7ffff46b6040) at /home/emaisin/src/binutils-gdb/gdb/python/python-internal.h:192 #14 0x000000000063fa33 in gdbpy_ref_policy::decref (ptr=0x7ffff46b6040) at /home/emaisin/src/binutils-gdb/gdb/python/py-ref.h:35 #15 0x000000000063fa77 in gdb::ref_ptr<_object, gdbpy_ref_policy>::~ref_ptr (this=0x7fffffffcdf0, __in_chrg=<optimized out>) at /home/emaisin/src/binutils-gdb/gdb/common/gdb_ref_ptr.h:91 #16 0x000000000064d8b8 in fnpy_call (gdbarch=0x2b50010, language=0x115d2c0 <c_language_defn>, cookie=0x7ffff46b7468, argc=1, argv=0x7fffffffcf48) at /home/emaisin/src/binutils-gdb/gdb/python/py-function.c:145 The fix is to place the gdbpy_enter first in the function. I also cleaned up the comments a bit and removed the unnecessary initialization of the value variable. gdb/ChangeLog: * python/py-function.c (fnpy_call): Reorder declarations to have the gdbpy_enter object declared first. * python/py-xmethods.c (gdbpy_get_xmethod_arg_types): Likewise.
657 lines
17 KiB
C
657 lines
17 KiB
C
/* Support for debug methods in Python.
|
|
|
|
Copyright (C) 2013-2017 Free Software Foundation, Inc.
|
|
|
|
This file is part of GDB.
|
|
|
|
This program 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.
|
|
|
|
This program 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, see <http://www.gnu.org/licenses/>. */
|
|
|
|
#include "defs.h"
|
|
#include "arch-utils.h"
|
|
#include "extension-priv.h"
|
|
#include "objfiles.h"
|
|
#include "value.h"
|
|
#include "language.h"
|
|
|
|
#include "python.h"
|
|
#include "python-internal.h"
|
|
#include "py-ref.h"
|
|
|
|
static const char enabled_field_name[] = "enabled";
|
|
static const char match_method_name[] = "match";
|
|
static const char get_arg_types_method_name[] = "get_arg_types";
|
|
static const char get_result_type_method_name[] = "get_result_type";
|
|
static const char matchers_attr_str[] = "xmethods";
|
|
|
|
static PyObject *py_match_method_name = NULL;
|
|
static PyObject *py_get_arg_types_method_name = NULL;
|
|
|
|
struct gdbpy_worker_data
|
|
{
|
|
PyObject *worker;
|
|
PyObject *this_type;
|
|
};
|
|
|
|
static struct xmethod_worker *new_python_xmethod_worker (PyObject *item,
|
|
PyObject *py_obj_type);
|
|
|
|
/* Implementation of free_xmethod_worker_data for Python. */
|
|
|
|
void
|
|
gdbpy_free_xmethod_worker_data (const struct extension_language_defn *extlang,
|
|
void *data)
|
|
{
|
|
struct gdbpy_worker_data *worker_data = (struct gdbpy_worker_data *) data;
|
|
|
|
gdb_assert (worker_data->worker != NULL && worker_data->this_type != NULL);
|
|
|
|
/* We don't do much here, but we still need the GIL. */
|
|
gdbpy_enter enter_py (get_current_arch (), current_language);
|
|
|
|
Py_DECREF (worker_data->worker);
|
|
Py_DECREF (worker_data->this_type);
|
|
xfree (worker_data);
|
|
}
|
|
|
|
/* Implementation of clone_xmethod_worker_data for Python. */
|
|
|
|
void *
|
|
gdbpy_clone_xmethod_worker_data (const struct extension_language_defn *extlang,
|
|
void *data)
|
|
{
|
|
struct gdbpy_worker_data *worker_data
|
|
= (struct gdbpy_worker_data *) data, *new_data;
|
|
|
|
gdb_assert (worker_data->worker != NULL && worker_data->this_type != NULL);
|
|
|
|
/* We don't do much here, but we still need the GIL. */
|
|
gdbpy_enter enter_py (get_current_arch (), current_language);
|
|
|
|
new_data = XCNEW (struct gdbpy_worker_data);
|
|
new_data->worker = worker_data->worker;
|
|
new_data->this_type = worker_data->this_type;
|
|
Py_INCREF (new_data->worker);
|
|
Py_INCREF (new_data->this_type);
|
|
|
|
return new_data;
|
|
}
|
|
|
|
/* Invoke the "match" method of the MATCHER and return a new reference
|
|
to the result. Returns NULL on error. */
|
|
|
|
static PyObject *
|
|
invoke_match_method (PyObject *matcher, PyObject *py_obj_type,
|
|
const char *xmethod_name)
|
|
{
|
|
int enabled;
|
|
|
|
gdbpy_ref enabled_field (PyObject_GetAttrString (matcher,
|
|
enabled_field_name));
|
|
if (enabled_field == NULL)
|
|
return NULL;
|
|
|
|
enabled = PyObject_IsTrue (enabled_field.get ());
|
|
if (enabled == -1)
|
|
return NULL;
|
|
if (enabled == 0)
|
|
{
|
|
/* Return 'None' if the matcher is not enabled. */
|
|
Py_RETURN_NONE;
|
|
}
|
|
|
|
gdbpy_ref match_method (PyObject_GetAttrString (matcher, match_method_name));
|
|
if (match_method == NULL)
|
|
return NULL;
|
|
|
|
gdbpy_ref py_xmethod_name (PyString_FromString (xmethod_name));
|
|
if (py_xmethod_name == NULL)
|
|
return NULL;
|
|
|
|
return PyObject_CallMethodObjArgs (matcher, py_match_method_name,
|
|
py_obj_type, py_xmethod_name.get (),
|
|
NULL);
|
|
}
|
|
|
|
/* Implementation of get_matching_xmethod_workers for Python. */
|
|
|
|
enum ext_lang_rc
|
|
gdbpy_get_matching_xmethod_workers
|
|
(const struct extension_language_defn *extlang,
|
|
struct type *obj_type, const char *method_name,
|
|
xmethod_worker_vec **dm_vec)
|
|
{
|
|
struct objfile *objfile;
|
|
VEC (xmethod_worker_ptr) *worker_vec = NULL;
|
|
PyObject *py_progspace;
|
|
|
|
gdb_assert (obj_type != NULL && method_name != NULL);
|
|
|
|
gdbpy_enter enter_py (get_current_arch (), current_language);
|
|
|
|
gdbpy_ref py_type (type_to_type_object (obj_type));
|
|
if (py_type == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
|
|
/* Create an empty list of debug methods. */
|
|
gdbpy_ref py_xmethod_matcher_list (PyList_New (0));
|
|
if (py_xmethod_matcher_list == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
|
|
/* Gather debug method matchers registered with the object files.
|
|
This could be done differently by iterating over each objfile's matcher
|
|
list individually, but there's no data yet to show it's needed. */
|
|
ALL_OBJFILES (objfile)
|
|
{
|
|
PyObject *py_objfile = objfile_to_objfile_object (objfile);
|
|
|
|
if (py_objfile == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
|
|
gdbpy_ref objfile_matchers (objfpy_get_xmethods (py_objfile, NULL));
|
|
gdbpy_ref temp (PySequence_Concat (py_xmethod_matcher_list.get (),
|
|
objfile_matchers.get ()));
|
|
if (temp == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
|
|
py_xmethod_matcher_list = std::move (temp);
|
|
}
|
|
|
|
/* Gather debug methods matchers registered with the current program
|
|
space. */
|
|
py_progspace = pspace_to_pspace_object (current_program_space);
|
|
if (py_progspace != NULL)
|
|
{
|
|
gdbpy_ref pspace_matchers (pspy_get_xmethods (py_progspace, NULL));
|
|
|
|
gdbpy_ref temp (PySequence_Concat (py_xmethod_matcher_list.get (),
|
|
pspace_matchers.get ()));
|
|
if (temp == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
|
|
py_xmethod_matcher_list = std::move (temp);
|
|
}
|
|
else
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
|
|
/* Gather debug method matchers registered globally. */
|
|
if (gdb_python_module != NULL
|
|
&& PyObject_HasAttrString (gdb_python_module, matchers_attr_str))
|
|
{
|
|
gdbpy_ref gdb_matchers (PyObject_GetAttrString (gdb_python_module,
|
|
matchers_attr_str));
|
|
if (gdb_matchers != NULL)
|
|
{
|
|
gdbpy_ref temp (PySequence_Concat (py_xmethod_matcher_list.get (),
|
|
gdb_matchers.get ()));
|
|
if (temp == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
|
|
py_xmethod_matcher_list = std::move (temp);
|
|
}
|
|
else
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
}
|
|
|
|
gdbpy_ref list_iter (PyObject_GetIter (py_xmethod_matcher_list.get ()));
|
|
if (list_iter == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
while (true)
|
|
{
|
|
gdbpy_ref matcher (PyIter_Next (list_iter.get ()));
|
|
if (matcher == NULL)
|
|
{
|
|
if (PyErr_Occurred ())
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
break;
|
|
}
|
|
|
|
gdbpy_ref match_result (invoke_match_method (matcher.get (),
|
|
py_type.get (),
|
|
method_name));
|
|
|
|
if (match_result == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
if (match_result == Py_None)
|
|
; /* This means there was no match. */
|
|
else if (PySequence_Check (match_result.get ()))
|
|
{
|
|
gdbpy_ref iter (PyObject_GetIter (match_result.get ()));
|
|
|
|
if (iter == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
while (true)
|
|
{
|
|
struct xmethod_worker *worker;
|
|
|
|
gdbpy_ref py_worker (PyIter_Next (iter.get ()));
|
|
if (py_worker == NULL)
|
|
{
|
|
if (PyErr_Occurred ())
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
break;
|
|
}
|
|
|
|
worker = new_python_xmethod_worker (py_worker.get (),
|
|
py_type.get ());
|
|
VEC_safe_push (xmethod_worker_ptr, worker_vec, worker);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
struct xmethod_worker *worker;
|
|
|
|
worker = new_python_xmethod_worker (match_result.get (),
|
|
py_type.get ());
|
|
VEC_safe_push (xmethod_worker_ptr, worker_vec, worker);
|
|
}
|
|
}
|
|
|
|
*dm_vec = worker_vec;
|
|
|
|
return EXT_LANG_RC_OK;
|
|
}
|
|
|
|
/* Implementation of get_xmethod_arg_types for Python. */
|
|
|
|
enum ext_lang_rc
|
|
gdbpy_get_xmethod_arg_types (const struct extension_language_defn *extlang,
|
|
struct xmethod_worker *worker,
|
|
int *nargs, struct type ***arg_types)
|
|
{
|
|
/* The gdbpy_enter object needs to be placed first, so that it's the last to
|
|
be destroyed. */
|
|
gdbpy_enter enter_py (get_current_arch (), current_language);
|
|
struct gdbpy_worker_data *worker_data
|
|
= (struct gdbpy_worker_data *) worker->data;
|
|
PyObject *py_worker = worker_data->worker;
|
|
struct type *obj_type;
|
|
int i = 1, arg_count;
|
|
gdbpy_ref list_iter;
|
|
|
|
/* Set nargs to -1 so that any premature return from this function returns
|
|
an invalid/unusable number of arg types. */
|
|
*nargs = -1;
|
|
|
|
gdbpy_ref get_arg_types_method
|
|
(PyObject_GetAttrString (py_worker, get_arg_types_method_name));
|
|
if (get_arg_types_method == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
|
|
gdbpy_ref py_argtype_list
|
|
(PyObject_CallMethodObjArgs (py_worker, py_get_arg_types_method_name,
|
|
NULL));
|
|
if (py_argtype_list == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
|
|
if (py_argtype_list == Py_None)
|
|
arg_count = 0;
|
|
else if (PySequence_Check (py_argtype_list.get ()))
|
|
{
|
|
arg_count = PySequence_Size (py_argtype_list.get ());
|
|
if (arg_count == -1)
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
|
|
list_iter.reset (PyObject_GetIter (py_argtype_list.get ()));
|
|
if (list_iter == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
}
|
|
else
|
|
arg_count = 1;
|
|
|
|
/* Include the 'this' argument in the size. */
|
|
gdb::unique_xmalloc_ptr<struct type *> type_array
|
|
(XCNEWVEC (struct type *, arg_count + 1));
|
|
i = 1;
|
|
if (list_iter != NULL)
|
|
{
|
|
while (true)
|
|
{
|
|
gdbpy_ref item (PyIter_Next (list_iter.get ()));
|
|
if (item == NULL)
|
|
{
|
|
if (PyErr_Occurred ())
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
break;
|
|
}
|
|
|
|
struct type *arg_type = type_object_to_type (item.get ());
|
|
if (arg_type == NULL)
|
|
{
|
|
PyErr_SetString (PyExc_TypeError,
|
|
_("Arg type returned by the get_arg_types "
|
|
"method of a debug method worker object is "
|
|
"not a gdb.Type object."));
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
|
|
(type_array.get ())[i] = arg_type;
|
|
i++;
|
|
}
|
|
}
|
|
else if (arg_count == 1)
|
|
{
|
|
/* py_argtype_list is not actually a list but a single gdb.Type
|
|
object. */
|
|
struct type *arg_type = type_object_to_type (py_argtype_list.get ());
|
|
|
|
if (arg_type == NULL)
|
|
{
|
|
PyErr_SetString (PyExc_TypeError,
|
|
_("Arg type returned by the get_arg_types method "
|
|
"of an xmethod worker object is not a gdb.Type "
|
|
"object."));
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
else
|
|
{
|
|
(type_array.get ())[i] = arg_type;
|
|
i++;
|
|
}
|
|
}
|
|
|
|
/* Add the type of 'this' as the first argument. The 'this' pointer should
|
|
be a 'const' value. Hence, create a 'const' variant of the 'this' pointer
|
|
type. */
|
|
obj_type = type_object_to_type (worker_data->this_type);
|
|
(type_array.get ())[0] = make_cv_type (1, 0, lookup_pointer_type (obj_type),
|
|
NULL);
|
|
*nargs = i;
|
|
*arg_types = type_array.release ();
|
|
|
|
return EXT_LANG_RC_OK;
|
|
}
|
|
|
|
/* Implementation of get_xmethod_result_type for Python. */
|
|
|
|
enum ext_lang_rc
|
|
gdbpy_get_xmethod_result_type (const struct extension_language_defn *extlang,
|
|
struct xmethod_worker *worker,
|
|
struct value *obj,
|
|
struct value **args, int nargs,
|
|
struct type **result_type_ptr)
|
|
{
|
|
struct gdbpy_worker_data *worker_data
|
|
= (struct gdbpy_worker_data *) worker->data;
|
|
PyObject *py_worker = worker_data->worker;
|
|
struct type *obj_type, *this_type;
|
|
int i;
|
|
|
|
gdbpy_enter enter_py (get_current_arch (), current_language);
|
|
|
|
/* First see if there is a get_result_type method.
|
|
If not this could be an old xmethod (pre 7.9.1). */
|
|
gdbpy_ref get_result_type_method
|
|
(PyObject_GetAttrString (py_worker, get_result_type_method_name));
|
|
if (get_result_type_method == NULL)
|
|
{
|
|
PyErr_Clear ();
|
|
*result_type_ptr = NULL;
|
|
return EXT_LANG_RC_OK;
|
|
}
|
|
|
|
obj_type = check_typedef (value_type (obj));
|
|
this_type = check_typedef (type_object_to_type (worker_data->this_type));
|
|
if (TYPE_CODE (obj_type) == TYPE_CODE_PTR)
|
|
{
|
|
struct type *this_ptr = lookup_pointer_type (this_type);
|
|
|
|
if (!types_equal (obj_type, this_ptr))
|
|
obj = value_cast (this_ptr, obj);
|
|
}
|
|
else if (TYPE_CODE (obj_type) == TYPE_CODE_REF)
|
|
{
|
|
struct type *this_ref = lookup_reference_type (this_type);
|
|
|
|
if (!types_equal (obj_type, this_ref))
|
|
obj = value_cast (this_ref, obj);
|
|
}
|
|
else
|
|
{
|
|
if (!types_equal (obj_type, this_type))
|
|
obj = value_cast (this_type, obj);
|
|
}
|
|
gdbpy_ref py_value_obj (value_to_value_object (obj));
|
|
if (py_value_obj == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
|
|
gdbpy_ref py_arg_tuple (PyTuple_New (nargs + 1));
|
|
if (py_arg_tuple == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
|
|
/* PyTuple_SET_ITEM steals the reference of the element, hence the
|
|
release. */
|
|
PyTuple_SET_ITEM (py_arg_tuple.get (), 0, py_value_obj.release ());
|
|
|
|
for (i = 0; i < nargs; i++)
|
|
{
|
|
PyObject *py_value_arg = value_to_value_object (args[i]);
|
|
|
|
if (py_value_arg == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
PyTuple_SET_ITEM (py_arg_tuple.get (), i + 1, py_value_arg);
|
|
}
|
|
|
|
gdbpy_ref py_result_type
|
|
(PyObject_CallObject (get_result_type_method.get (), py_arg_tuple.get ()));
|
|
if (py_result_type == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
|
|
*result_type_ptr = type_object_to_type (py_result_type.get ());
|
|
if (*result_type_ptr == NULL)
|
|
{
|
|
PyErr_SetString (PyExc_TypeError,
|
|
_("Type returned by the get_result_type method of an"
|
|
" xmethod worker object is not a gdb.Type object."));
|
|
gdbpy_print_stack ();
|
|
return EXT_LANG_RC_ERROR;
|
|
}
|
|
|
|
return EXT_LANG_RC_OK;
|
|
}
|
|
|
|
/* Implementation of invoke_xmethod for Python. */
|
|
|
|
struct value *
|
|
gdbpy_invoke_xmethod (const struct extension_language_defn *extlang,
|
|
struct xmethod_worker *worker,
|
|
struct value *obj, struct value **args, int nargs)
|
|
{
|
|
int i;
|
|
struct type *obj_type, *this_type;
|
|
struct value *res = NULL;
|
|
struct gdbpy_worker_data *worker_data
|
|
= (struct gdbpy_worker_data *) worker->data;
|
|
PyObject *xmethod_worker = worker_data->worker;
|
|
|
|
gdbpy_enter enter_py (get_current_arch (), current_language);
|
|
|
|
obj_type = check_typedef (value_type (obj));
|
|
this_type = check_typedef (type_object_to_type (worker_data->this_type));
|
|
if (TYPE_CODE (obj_type) == TYPE_CODE_PTR)
|
|
{
|
|
struct type *this_ptr = lookup_pointer_type (this_type);
|
|
|
|
if (!types_equal (obj_type, this_ptr))
|
|
obj = value_cast (this_ptr, obj);
|
|
}
|
|
else if (TYPE_CODE (obj_type) == TYPE_CODE_REF)
|
|
{
|
|
struct type *this_ref = lookup_reference_type (this_type);
|
|
|
|
if (!types_equal (obj_type, this_ref))
|
|
obj = value_cast (this_ref, obj);
|
|
}
|
|
else
|
|
{
|
|
if (!types_equal (obj_type, this_type))
|
|
obj = value_cast (this_type, obj);
|
|
}
|
|
gdbpy_ref py_value_obj (value_to_value_object (obj));
|
|
if (py_value_obj == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
error (_("Error while executing Python code."));
|
|
}
|
|
|
|
gdbpy_ref py_arg_tuple (PyTuple_New (nargs + 1));
|
|
if (py_arg_tuple == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
error (_("Error while executing Python code."));
|
|
}
|
|
|
|
/* PyTuple_SET_ITEM steals the reference of the element, hence the
|
|
release. */
|
|
PyTuple_SET_ITEM (py_arg_tuple.get (), 0, py_value_obj.release ());
|
|
|
|
for (i = 0; i < nargs; i++)
|
|
{
|
|
PyObject *py_value_arg = value_to_value_object (args[i]);
|
|
|
|
if (py_value_arg == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
error (_("Error while executing Python code."));
|
|
}
|
|
|
|
PyTuple_SET_ITEM (py_arg_tuple.get (), i + 1, py_value_arg);
|
|
}
|
|
|
|
gdbpy_ref py_result (PyObject_CallObject (xmethod_worker,
|
|
py_arg_tuple.get ()));
|
|
if (py_result == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
error (_("Error while executing Python code."));
|
|
}
|
|
|
|
if (py_result != Py_None)
|
|
{
|
|
res = convert_value_from_python (py_result.get ());
|
|
if (res == NULL)
|
|
{
|
|
gdbpy_print_stack ();
|
|
error (_("Error while executing Python code."));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
res = allocate_value (lookup_typename (python_language, python_gdbarch,
|
|
"void", NULL, 0));
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
/* Creates a new Python xmethod_worker object.
|
|
The new object has data of type 'struct gdbpy_worker_data' composed
|
|
with the components PY_WORKER and THIS_TYPE. */
|
|
|
|
static struct xmethod_worker *
|
|
new_python_xmethod_worker (PyObject *py_worker, PyObject *this_type)
|
|
{
|
|
struct gdbpy_worker_data *data;
|
|
|
|
gdb_assert (py_worker != NULL && this_type != NULL);
|
|
|
|
data = XCNEW (struct gdbpy_worker_data);
|
|
data->worker = py_worker;
|
|
data->this_type = this_type;
|
|
Py_INCREF (py_worker);
|
|
Py_INCREF (this_type);
|
|
|
|
return new_xmethod_worker (&extension_language_python, data);
|
|
}
|
|
|
|
int
|
|
gdbpy_initialize_xmethods (void)
|
|
{
|
|
py_match_method_name = PyString_FromString (match_method_name);
|
|
if (py_match_method_name == NULL)
|
|
return -1;
|
|
|
|
py_get_arg_types_method_name
|
|
= PyString_FromString (get_arg_types_method_name);
|
|
if (py_get_arg_types_method_name == NULL)
|
|
return -1;
|
|
|
|
return 1;
|
|
}
|