- /* Convenience functions implemented in Python.
 
 
-    Copyright (C) 2008-2015 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 "value.h"
 
- #include "python-internal.h"
 
- #include "charset.h"
 
- #include "gdbcmd.h"
 
- #include "cli/cli-decode.h"
 
- #include "completer.h"
 
- #include "expression.h"
 
- #include "language.h"
 
 
- static PyTypeObject fnpy_object_type
 
-     CPYCHECKER_TYPE_OBJECT_FOR_TYPEDEF ("PyObject");
 
 
 
 
- static PyObject *
 
- convert_values_to_python (int argc, struct value **argv)
 
- {
 
-   int i;
 
-   PyObject *result = PyTuple_New (argc);
 
 
-   if (! result)
 
-     return NULL;
 
 
-   for (i = 0; i < argc; ++i)
 
-     {
 
-       PyObject *elt = value_to_value_object (argv[i]);
 
-       if (! elt)
 
-         {
 
-           Py_DECREF (result);
 
-           return NULL;
 
-         }
 
-       PyTuple_SetItem (result, i, elt);
 
-     }
 
-   return result;
 
- }
 
 
- /* Call a Python function object's invoke method.  */
 
 
- static struct value *
 
- fnpy_call (struct gdbarch *gdbarch, const struct language_defn *language,
 
-            void *cookie, int argc, struct value **argv)
 
- {
 
-   struct value *value = NULL;
 
-   /* 'result' must be set to NULL, this initially indicates whether
 
-      the function was called, or not.  */
 
-   PyObject *result = NULL;
 
-   PyObject *callable, *args;
 
-   struct cleanup *cleanup;
 
 
-   cleanup = ensure_python_env (gdbarch, language);
 
 
-   args = convert_values_to_python (argc, argv);
 
-   /* convert_values_to_python can return NULL on error.  If we
 
-      encounter this, do not call the function, but allow the Python ->
 
-      error code conversion below to deal with the Python exception.
 
-      Note, that this is different if the function simply does not
 
-      have arguments.  */
 
 
-   if (args)
 
-     {
 
-       callable = PyObject_GetAttrString ((PyObject *) cookie, "invoke");
 
-       if (! callable)
 
-         {
 
-           Py_DECREF (args);
 
-           error (_("No method named 'invoke' in object."));
 
-         }
 
 
-       result = PyObject_Call (callable, args, NULL);
 
-       Py_DECREF (callable);
 
-       Py_DECREF (args);
 
-     }
 
 
-   if (!result)
 
-     {
 
-       PyObject *ptype, *pvalue, *ptraceback;
 
-       char *msg;
 
 
-       PyErr_Fetch (&ptype, &pvalue, &ptraceback);
 
 
-       /* Try to fetch an error message contained within ptype, pvalue.
 
-          When fetching the error message we need to make our own copy,
 
-          we no longer own ptype, pvalue after the call to PyErr_Restore.  */
 
 
-       msg = gdbpy_exception_to_string (ptype, pvalue);
 
-       make_cleanup (xfree, msg);
 
 
-       if (msg == NULL)
 
-         {
 
-           /* An error occurred computing the string representation of the
 
-              error message.  This is rare, but we should inform the user.  */
 
 
-           printf_filtered (_("An error occurred in a Python "
 
-                              "convenience function\n"
 
-                              "and then another occurred computing the "
 
-                              "error message.\n"));
 
-           gdbpy_print_stack ();
 
-         }
 
 
-       /* Don't print the stack for gdb.GdbError exceptions.
 
-          It is generally used to flag user errors.
 
 
-          We also don't want to print "Error occurred in Python command"
 
-          for user errors.  However, a missing message for gdb.GdbError
 
-          exceptions is arguably a bug, so we flag it as such.  */
 
 
-       if (!PyErr_GivenExceptionMatches (ptype, gdbpy_gdberror_exc)
 
-           || msg == NULL || *msg == '\0')
 
-         {
 
-           PyErr_Restore (ptype, pvalue, ptraceback);
 
-           gdbpy_print_stack ();
 
-           if (msg != NULL && *msg != '\0')
 
-             error (_("Error occurred in Python convenience function: %s"),
 
-                    msg);
 
-           else
 
-             error (_("Error occurred in Python convenience function."));
 
-         }
 
-       else
 
-         {
 
-           Py_XDECREF (ptype);
 
-           Py_XDECREF (pvalue);
 
-           Py_XDECREF (ptraceback);
 
-           error ("%s", msg);
 
-         }
 
-     }
 
 
-   value = convert_value_from_python (result);
 
-   if (value == NULL)
 
-     {
 
-       Py_DECREF (result);
 
-       gdbpy_print_stack ();
 
-       error (_("Error while executing Python code."));
 
-     }
 
 
-   Py_DECREF (result);
 
-   do_cleanups (cleanup);
 
 
-   return value;
 
- }
 
 
- /* Initializer for a Function object.  It takes one argument, the name
 
-    of the function.  */
 
 
- static int
 
- fnpy_init (PyObject *self, PyObject *args, PyObject *kwds)
 
- {
 
-   const char *name;
 
-   char *docstring = NULL;
 
 
-   if (! PyArg_ParseTuple (args, "s", &name))
 
-     return -1;
 
-   Py_INCREF (self);
 
 
-   if (PyObject_HasAttrString (self, "__doc__"))
 
-     {
 
-       PyObject *ds_obj = PyObject_GetAttrString (self, "__doc__");
 
-       if (ds_obj != NULL)
 
-         {
 
-           if (gdbpy_is_string (ds_obj))
 
-             {
 
-               docstring = python_string_to_host_string (ds_obj);
 
-               if (docstring == NULL)
 
-                 {
 
-                   Py_DECREF (self);
 
-                   Py_DECREF (ds_obj);
 
-                   return -1;
 
-                 }
 
-             }
 
 
-           Py_DECREF (ds_obj);
 
-         }
 
-     }
 
-   if (! docstring)
 
-     docstring = xstrdup (_("This function is not documented."));
 
 
-   add_internal_function (name, docstring, fnpy_call, self);
 
-   return 0;
 
- }
 
 
- /* Initialize internal function support.  */
 
 
- int
 
- gdbpy_initialize_functions (void)
 
- {
 
-   fnpy_object_type.tp_new = PyType_GenericNew;
 
-   if (PyType_Ready (&fnpy_object_type) < 0)
 
-     return -1;
 
 
-   return gdb_pymodule_addobject (gdb_module, "Function",
 
-                                  (PyObject *) &fnpy_object_type);
 
- }
 
 
 
 
- static PyTypeObject fnpy_object_type =
 
- {
 
-   PyVarObject_HEAD_INIT (NULL, 0)
 
-   "gdb.Function",                  /*tp_name*/
 
-   sizeof (PyObject),                  /*tp_basicsize*/
 
-   0,                                  /*tp_itemsize*/
 
-   0,                                  /*tp_dealloc*/
 
-   0,                                  /*tp_print*/
 
-   0,                                  /*tp_getattr*/
 
-   0,                                  /*tp_setattr*/
 
-   0,                                  /*tp_compare*/
 
-   0,                                  /*tp_repr*/
 
-   0,                                  /*tp_as_number*/
 
-   0,                                  /*tp_as_sequence*/
 
-   0,                                  /*tp_as_mapping*/
 
-   0,                                  /*tp_hash */
 
-   0,                                  /*tp_call*/
 
-   0,                                  /*tp_str*/
 
-   0,                                  /*tp_getattro*/
 
-   0,                                  /*tp_setattro*/
 
-   0,                                  /*tp_as_buffer*/
 
-   Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE, /*tp_flags*/
 
-   "GDB function object",          /* tp_doc */
 
-   0,                                  /* tp_traverse */
 
-   0,                                  /* tp_clear */
 
-   0,                                  /* tp_richcompare */
 
-   0,                                  /* tp_weaklistoffset */
 
-   0,                                  /* tp_iter */
 
-   0,                                  /* tp_iternext */
 
-   0,                                  /* tp_methods */
 
-   0,                                  /* tp_members */
 
-   0,                                  /* tp_getset */
 
-   0,                                  /* tp_base */
 
-   0,                                  /* tp_dict */
 
-   0,                                  /* tp_descr_get */
 
-   0,                                  /* tp_descr_set */
 
-   0,                                  /* tp_dictoffset */
 
-   fnpy_init,                          /* tp_init */
 
-   0,                                  /* tp_alloc */
 
- };