Iterate over 'struct varobj_item' instead of PyObject
In previous patch, "saved_item" is still a PyOjbect and iteration is still performed over PyObject. This patch continues to decouple iteration from python code, so it changes its type to "struct varobj_item *", so that the iterator itself is independent of python. V2: - Call varobj_delete_iter in free_variable. - Fix changelog entries. - Use XNEW. V3: - Return NULL early in py_varobj_iter_next if gdb_python_initialized is false. gdb: 2014-06-12 Pedro Alves <pedro@codesourcery.com> Yao Qi <yao@codesourcery.com> * python/py-varobj.c (py_varobj_iter_next): Return NULL if gdb_python_initialized is false. Move some code from varobj.c. * varobj-iter.h (struct varobj_item): Moved from varobj.c. * varobj.c: Move "varobj-iter.h" inclusion earlier. (struct varobj_item): Moved to varobj-iter.h". (varobj_clear_saved_item): New function. (update_dynamic_varobj_children): Move python-related code to py-varobj.c. (free_variable): Call varobj_clear_saved_item and varobj_iter_delete.
This commit is contained in:
parent
e525021603
commit
827f100cee
4 changed files with 70 additions and 53 deletions
|
@ -1,3 +1,17 @@
|
||||||
|
2014-06-12 Pedro Alves <pedro@codesourcery.com>
|
||||||
|
Yao Qi <yao@codesourcery.com>
|
||||||
|
|
||||||
|
* python/py-varobj.c (py_varobj_iter_next): Return NULL if
|
||||||
|
gdb_python_initialized is false. Move some code from varobj.c.
|
||||||
|
* varobj-iter.h (struct varobj_item): Moved from varobj.c.
|
||||||
|
* varobj.c: Move "varobj-iter.h" inclusion earlier.
|
||||||
|
(struct varobj_item): Moved to varobj-iter.h".
|
||||||
|
(varobj_clear_saved_item): New function.
|
||||||
|
(update_dynamic_varobj_children): Move python-related code to
|
||||||
|
py-varobj.c.
|
||||||
|
(free_variable): Call varobj_clear_saved_item and
|
||||||
|
varobj_iter_delete.
|
||||||
|
|
||||||
2014-06-12 Pedro Alves <pedro@codesourcery.com>
|
2014-06-12 Pedro Alves <pedro@codesourcery.com>
|
||||||
Yao Qi <yao@codesourcery.com>
|
Yao Qi <yao@codesourcery.com>
|
||||||
|
|
||||||
|
|
|
@ -54,6 +54,12 @@ py_varobj_iter_next (struct varobj_iter *self)
|
||||||
struct py_varobj_iter *t = (struct py_varobj_iter *) self;
|
struct py_varobj_iter *t = (struct py_varobj_iter *) self;
|
||||||
struct cleanup *back_to;
|
struct cleanup *back_to;
|
||||||
PyObject *item;
|
PyObject *item;
|
||||||
|
PyObject *py_v;
|
||||||
|
varobj_item *vitem;
|
||||||
|
const char *name = NULL;
|
||||||
|
|
||||||
|
if (!gdb_python_initialized)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
back_to = varobj_ensure_python_env (self->var);
|
back_to = varobj_ensure_python_env (self->var);
|
||||||
|
|
||||||
|
@ -101,9 +107,21 @@ py_varobj_iter_next (struct varobj_iter *self)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!PyArg_ParseTuple (item, "sO", &name, &py_v))
|
||||||
|
{
|
||||||
|
gdbpy_print_stack ();
|
||||||
|
error (_("Invalid item from the child list"));
|
||||||
|
}
|
||||||
|
|
||||||
|
vitem = xmalloc (sizeof *vitem);
|
||||||
|
vitem->value = convert_value_from_python (py_v);
|
||||||
|
if (vitem->value == NULL)
|
||||||
|
gdbpy_print_stack ();
|
||||||
|
vitem->name = xstrdup (name);
|
||||||
|
|
||||||
self->next_raw_index++;
|
self->next_raw_index++;
|
||||||
do_cleanups (back_to);
|
do_cleanups (back_to);
|
||||||
return item;
|
return vitem;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* The 'vtable' of pretty-printed python varobj iterators. */
|
/* The 'vtable' of pretty-printed python varobj iterators. */
|
||||||
|
|
|
@ -14,9 +14,18 @@
|
||||||
You should have received a copy of the GNU General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
||||||
|
|
||||||
struct varobj_iter_ops;
|
/* A node or item of varobj, composed of the name and the value. */
|
||||||
|
|
||||||
typedef PyObject varobj_item;
|
typedef struct varobj_item
|
||||||
|
{
|
||||||
|
/* Name of this item. */
|
||||||
|
char *name;
|
||||||
|
|
||||||
|
/* Value of this item. */
|
||||||
|
struct value *value;
|
||||||
|
} varobj_item;
|
||||||
|
|
||||||
|
struct varobj_iter_ops;
|
||||||
|
|
||||||
/* A dynamic varobj iterator "class". */
|
/* A dynamic varobj iterator "class". */
|
||||||
|
|
||||||
|
|
76
gdb/varobj.c
76
gdb/varobj.c
|
@ -33,6 +33,7 @@
|
||||||
#include "vec.h"
|
#include "vec.h"
|
||||||
#include "gdbthread.h"
|
#include "gdbthread.h"
|
||||||
#include "inferior.h"
|
#include "inferior.h"
|
||||||
|
#include "varobj-iter.h"
|
||||||
|
|
||||||
#if HAVE_PYTHON
|
#if HAVE_PYTHON
|
||||||
#include "python/python.h"
|
#include "python/python.h"
|
||||||
|
@ -41,8 +42,6 @@
|
||||||
typedef int PyObject;
|
typedef int PyObject;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "varobj-iter.h"
|
|
||||||
|
|
||||||
/* Non-zero if we want to see trace of varobj level stuff. */
|
/* Non-zero if we want to see trace of varobj level stuff. */
|
||||||
|
|
||||||
unsigned int varobjdebug = 0;
|
unsigned int varobjdebug = 0;
|
||||||
|
@ -110,17 +109,6 @@ struct varobj_root
|
||||||
struct varobj_root *next;
|
struct varobj_root *next;
|
||||||
};
|
};
|
||||||
|
|
||||||
/* A node or item of varobj, composed of the name and the value. */
|
|
||||||
|
|
||||||
struct varobj_item
|
|
||||||
{
|
|
||||||
/* Name of this item. */
|
|
||||||
char *name;
|
|
||||||
|
|
||||||
/* Value of this item. */
|
|
||||||
struct value *value;
|
|
||||||
};
|
|
||||||
|
|
||||||
/* Dynamic part of varobj. */
|
/* Dynamic part of varobj. */
|
||||||
|
|
||||||
struct varobj_dynamic
|
struct varobj_dynamic
|
||||||
|
@ -788,6 +776,18 @@ varobj_get_iterator (struct varobj *var)
|
||||||
requested an iterator from a non-dynamic varobj"));
|
requested an iterator from a non-dynamic varobj"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Release and clear VAR's saved item, if any. */
|
||||||
|
|
||||||
|
static void
|
||||||
|
varobj_clear_saved_item (struct varobj_dynamic *var)
|
||||||
|
{
|
||||||
|
if (var->saved_item != NULL)
|
||||||
|
{
|
||||||
|
value_free (var->saved_item->value);
|
||||||
|
xfree (var->saved_item);
|
||||||
|
var->saved_item = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static int
|
static int
|
||||||
|
@ -802,14 +802,8 @@ update_dynamic_varobj_children (struct varobj *var,
|
||||||
int to)
|
int to)
|
||||||
{
|
{
|
||||||
#if HAVE_PYTHON
|
#if HAVE_PYTHON
|
||||||
struct cleanup *back_to;
|
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
if (!gdb_python_initialized)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
back_to = varobj_ensure_python_env (var);
|
|
||||||
|
|
||||||
*cchanged = 0;
|
*cchanged = 0;
|
||||||
|
|
||||||
if (update_children || var->dynamic->child_iter == NULL)
|
if (update_children || var->dynamic->child_iter == NULL)
|
||||||
|
@ -817,16 +811,12 @@ update_dynamic_varobj_children (struct varobj *var,
|
||||||
varobj_iter_delete (var->dynamic->child_iter);
|
varobj_iter_delete (var->dynamic->child_iter);
|
||||||
var->dynamic->child_iter = varobj_get_iterator (var);
|
var->dynamic->child_iter = varobj_get_iterator (var);
|
||||||
|
|
||||||
Py_XDECREF (var->dynamic->saved_item);
|
varobj_clear_saved_item (var->dynamic);
|
||||||
var->dynamic->saved_item = NULL;
|
|
||||||
|
|
||||||
i = 0;
|
i = 0;
|
||||||
|
|
||||||
if (var->dynamic->child_iter == NULL)
|
if (var->dynamic->child_iter == NULL)
|
||||||
{
|
return 0;
|
||||||
do_cleanups (back_to);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
i = VEC_length (varobj_p, var->children);
|
i = VEC_length (varobj_p, var->children);
|
||||||
|
@ -835,10 +825,10 @@ update_dynamic_varobj_children (struct varobj *var,
|
||||||
are more children. */
|
are more children. */
|
||||||
for (; to < 0 || i < to + 1; ++i)
|
for (; to < 0 || i < to + 1; ++i)
|
||||||
{
|
{
|
||||||
PyObject *item;
|
varobj_item *item;
|
||||||
|
|
||||||
/* See if there was a leftover from last time. */
|
/* See if there was a leftover from last time. */
|
||||||
if (var->dynamic->saved_item)
|
if (var->dynamic->saved_item != NULL)
|
||||||
{
|
{
|
||||||
item = var->dynamic->saved_item;
|
item = var->dynamic->saved_item;
|
||||||
var->dynamic->saved_item = NULL;
|
var->dynamic->saved_item = NULL;
|
||||||
|
@ -846,6 +836,10 @@ update_dynamic_varobj_children (struct varobj *var,
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
item = varobj_iter_next (var->dynamic->child_iter);
|
item = varobj_iter_next (var->dynamic->child_iter);
|
||||||
|
/* Release vitem->value so its lifetime is not bound to the
|
||||||
|
execution of a command. */
|
||||||
|
if (item != NULL && item->value != NULL)
|
||||||
|
release_value_or_incref (item->value);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (item == NULL)
|
if (item == NULL)
|
||||||
|
@ -858,36 +852,19 @@ update_dynamic_varobj_children (struct varobj *var,
|
||||||
/* We don't want to push the extra child on any report list. */
|
/* We don't want to push the extra child on any report list. */
|
||||||
if (to < 0 || i < to)
|
if (to < 0 || i < to)
|
||||||
{
|
{
|
||||||
PyObject *py_v;
|
|
||||||
const char *name;
|
|
||||||
struct varobj_item varobj_item;
|
|
||||||
struct cleanup *inner;
|
|
||||||
int can_mention = from < 0 || i >= from;
|
int can_mention = from < 0 || i >= from;
|
||||||
|
|
||||||
inner = make_cleanup_py_decref (item);
|
|
||||||
|
|
||||||
if (!PyArg_ParseTuple (item, "sO", &name, &py_v))
|
|
||||||
{
|
|
||||||
gdbpy_print_stack ();
|
|
||||||
error (_("Invalid item from the child list"));
|
|
||||||
}
|
|
||||||
|
|
||||||
varobj_item.value = convert_value_from_python (py_v);
|
|
||||||
if (varobj_item.value == NULL)
|
|
||||||
gdbpy_print_stack ();
|
|
||||||
varobj_item.name = xstrdup (name);
|
|
||||||
|
|
||||||
install_dynamic_child (var, can_mention ? changed : NULL,
|
install_dynamic_child (var, can_mention ? changed : NULL,
|
||||||
can_mention ? type_changed : NULL,
|
can_mention ? type_changed : NULL,
|
||||||
can_mention ? new : NULL,
|
can_mention ? new : NULL,
|
||||||
can_mention ? unchanged : NULL,
|
can_mention ? unchanged : NULL,
|
||||||
can_mention ? cchanged : NULL, i,
|
can_mention ? cchanged : NULL, i,
|
||||||
&varobj_item);
|
item);
|
||||||
do_cleanups (inner);
|
|
||||||
|
xfree (item);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
Py_XDECREF (var->dynamic->saved_item);
|
|
||||||
var->dynamic->saved_item = item;
|
var->dynamic->saved_item = item;
|
||||||
|
|
||||||
/* We want to truncate the child list just before this
|
/* We want to truncate the child list just before this
|
||||||
|
@ -913,7 +890,6 @@ update_dynamic_varobj_children (struct varobj *var,
|
||||||
|
|
||||||
var->num_children = VEC_length (varobj_p, var->children);
|
var->num_children = VEC_length (varobj_p, var->children);
|
||||||
|
|
||||||
do_cleanups (back_to);
|
|
||||||
return 1;
|
return 1;
|
||||||
#else
|
#else
|
||||||
gdb_assert_not_reached ("should never be called if Python is not enabled");
|
gdb_assert_not_reached ("should never be called if Python is not enabled");
|
||||||
|
@ -2191,12 +2167,12 @@ free_variable (struct varobj *var)
|
||||||
|
|
||||||
Py_XDECREF (var->dynamic->constructor);
|
Py_XDECREF (var->dynamic->constructor);
|
||||||
Py_XDECREF (var->dynamic->pretty_printer);
|
Py_XDECREF (var->dynamic->pretty_printer);
|
||||||
Py_XDECREF (var->dynamic->child_iter);
|
|
||||||
Py_XDECREF (var->dynamic->saved_item);
|
|
||||||
do_cleanups (cleanup);
|
do_cleanups (cleanup);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
varobj_iter_delete (var->dynamic->child_iter);
|
||||||
|
varobj_clear_saved_item (var->dynamic);
|
||||||
value_free (var->value);
|
value_free (var->value);
|
||||||
|
|
||||||
/* Free the expression if this is a root variable. */
|
/* Free the expression if this is a root variable. */
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue