From: Li Yu <raise.sail@gmail.com>
To: gdb-patches@sourceware.org
Cc: Paul Koning <paulkoning@comcast.net>
Subject: [PATCH v2] gdb/python: add missing handling for anonymous members of struct and union
Date: Fri, 30 Sep 2011 11:04:00 -0000 [thread overview]
Message-ID: <4E8595F6.7080004@gmail.com> (raw)
gdb.Type.fields() missed handling for anonymous members.
This patch fix it, below are details:
Assume that we have a c source as below:
/////////////////////////////
struct A
{
int a;
union {
int b0;
int b1;
union {
int bb0;
int bb1;
union {
int bbb0;
int bbb1;
};
};
};
int c;
union {
union {
int dd0;
int dd1;
};
int d2;
int d3;
};
};
int main()
{
struct A a;
}
////////////////////////////
And have a python gdb script as below:
##########################
v = gdb.parse_and_eval("a")
t = v.type
fields = t.fields()
for f in fields:
print "[%s]" % f.name, v[f.name]
##########################
Without this patch, above script will print:
[a] -7616
[] {{dd0 = 0, dd1 = 0}, d2 = 0, d3 = 0}
[c] 0
[] {{dd0 = 0, dd1 = 0}, d2 = 0, d3 = 0}
With this patch, above script will print rightly:
[a] -7616
[b0] 32767
[b1] 32767
[bb0] 32767
[bb1] 32767
[bbb0] 32767
[bbb1] 32767
[c] 0
[dd0] 0
[dd1] 0
[d2] 0
[d3] 0
And please note that this patch assumed that Paul's patch in
http://sourceware.org/ml/gdb-patches/2011-09/msg00546.html is
applied first.
Thanks for Paul's feedback for first version of this patch.
Signed-off-by: Li Yu <raise.sail@gmail.com>
gdb/python/:
2011-09-29 Li Yu <raise.sail@gmail.com>
* py-type.c: Add process for anonymous members of struct and union
py-type.c | 63 +++++++++++++++++++++++++++++++++++++++++++++++++++++---------
1 file changed, 54 insertions(+), 9 deletions(-)
diff --git a/gdb/python/py-type.c b/gdb/python/py-type.c
index 27f8b38..d0d8a94 100644
--- a/gdb/python/py-type.c
+++ b/gdb/python/py-type.c
@@ -56,15 +56,20 @@ typedef struct pyty_field_object
static PyTypeObject field_object_type;
/* A type iterator object. */
-typedef struct {
+struct __typy_iterator_object
+{
PyObject_HEAD
+ /* The iterators for support fields of anonymous field */
+ struct __typy_iterator_object *child;
+ struct __typy_iterator_object *parent;
/* The current field index. */
int field;
/* What to return. */
enum gdbpy_iter_kind kind;
/* Pointer back to the original source type object. */
struct pyty_type_object *source;
-} typy_iterator_object;
+};
+typedef struct __typy_iterator_object typy_iterator_object;
static PyTypeObject type_iterator_object_type;
@@ -1201,6 +1206,8 @@ typy_make_iter (PyObject *self, enum gdbpy_iter_kind kind)
if (typy_iter_obj == NULL)
return NULL;
+ typy_iter_obj->child = NULL;
+ typy_iter_obj->parent = NULL;
typy_iter_obj->field = 0;
typy_iter_obj->kind = kind;
Py_INCREF (self);
- struct type *type = iter_obj->source->type;
- int i;
- PyObject *result;
-
- if (iter_obj->field < TYPE_NFIELDS (type))
+ typy_iterator_object *iter_obj = (typy_iterator_object *) self, *child_iter_obj;
+ struct type *type;
+ PyObject *result, *child_pytype;
+ char *name;
+
+ while (iter_obj->child) /* deepest anonymous member first */
+ {
+ iter_obj = iter_obj->child;
+ }
+ type = iter_obj->source->type;
+
+restart:
+ while (iter_obj->field >= TYPE_NFIELDS (type))
+ {
+ iter_obj = iter_obj->parent;
+ if (!iter_obj)
+ return NULL;
+ Py_DECREF(iter_obj->child);
+ iter_obj->child = NULL;
+ type = iter_obj->source->type;
+ }
+
+ name = TYPE_FIELD_NAME (type, iter_obj->field);
+ if (!name)
+ goto abort_clean;
+
+ if (name[0]) /* mostly cases */
{
result = make_fielditem (type, iter_obj->field, iter_obj->kind);
if (result != NULL)
- iter_obj->field++;
+ iter_obj->field++;
return result;
}
+ /* handing for anonymous members here */
+ type = TYPE_FIELD_TYPE(type, iter_obj->field++);
+ child_pytype = type_to_type_object(type);
+ if (!child_pytype)
+ goto abort_clean;;
+ child_iter_obj = (typy_iterator_object*)typy_make_iter (child_pytype, iter_obj->kind);
+ iter_obj->child = child_iter_obj;
+ child_iter_obj->parent = iter_obj;
+ iter_obj = child_iter_obj;
+ goto restart;
+
+abort_clean:
+ while (iter_obj->parent)
+ {
+ iter_obj = iter_obj->parent;
+ Py_DECREF(iter_obj->child);
+ }
return NULL;
}
next reply other threads:[~2011-09-30 10:12 UTC|newest]
Thread overview: 21+ messages / expand[flat|nested] mbox.gz Atom feed top
2011-09-30 11:04 Li Yu [this message]
2011-09-30 16:15 ` Paul Koning
2011-10-01 14:01 ` Li Yu
2011-10-01 18:54 ` Paul Koning
2011-10-04 16:37 ` Tom Tromey
2011-10-04 18:05 ` Paul Koning
2011-10-04 20:24 ` Tom Tromey
2011-10-04 20:41 ` Paul Koning
2011-10-19 20:52 ` Tom Tromey
2011-10-19 20:59 ` Paul Koning
2011-10-20 18:49 ` Tom Tromey
2011-10-25 18:34 ` [RFA] Python: iterator for deep traversal of gdb.Type struct/union fields Paul Koning
2011-10-25 19:03 ` Paul Koning
2011-10-25 20:16 ` Eli Zaretskii
2011-10-26 17:14 ` Paul Koning
2011-10-27 13:01 ` Doug Evans
2011-10-27 14:52 ` Paul_Koning
2011-10-27 19:57 ` Tom Tromey
[not found] ` <09787EF419216C41A903FD14EE5506DD030CF168FB@AUSX7MCPC103.AMER.DELL.COM>
2011-10-27 21:56 ` Paul Koning
2011-10-27 22:14 ` Eli Zaretskii
2011-10-28 14:55 ` Paul Koning
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=4E8595F6.7080004@gmail.com \
--to=raise.sail@gmail.com \
--cc=gdb-patches@sourceware.org \
--cc=paulkoning@comcast.net \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox