Merged revisions 74917 via svnmerge from

svn+ssh://pythondev@svn.python.org/python/trunk

........
  r74917 | thomas.heller | 2009-09-18 20:55:17 +0200 (Fr, 18 Sep 2009) | 3 lines

  Issue #5042: Structure sub-subclass does now initialize correctly with
  base class positional arguments.
........

Also made small stylistic changes.
This commit is contained in:
Thomas Heller 2009-09-18 19:05:13 +00:00
parent a91ded2842
commit d7cb1b9119
3 changed files with 95 additions and 58 deletions

View File

@ -349,6 +349,25 @@ class StructureTestCase(unittest.TestCase):
self.assertTrue("from_address" in dir(type(Structure))) self.assertTrue("from_address" in dir(type(Structure)))
self.assertTrue("in_dll" in dir(type(Structure))) self.assertTrue("in_dll" in dir(type(Structure)))
def test_positional_args(self):
# see also http://bugs.python.org/issue5042
class W(Structure):
_fields_ = [("a", c_int), ("b", c_int)]
class X(W):
_fields_ = [("c", c_int)]
class Y(X):
pass
class Z(Y):
_fields_ = [("d", c_int), ("e", c_int), ("f", c_int)]
z = Z(1, 2, 3, 4, 5, 6)
self.assertEqual((z.a, z.b, z.c, z.d, z.e, z.f),
(1, 2, 3, 4, 5, 6))
z = Z(1)
self.assertEqual((z.a, z.b, z.c, z.d, z.e, z.f),
(1, 0, 0, 0, 0, 0))
self.assertRaises(TypeError, lambda: Z(1, 2, 3, 4, 5, 6, 7))
class PointerMemberTestCase(unittest.TestCase): class PointerMemberTestCase(unittest.TestCase):
def test(self): def test(self):

View File

@ -72,6 +72,9 @@ C-API
Library Library
------- -------
- Issue #5042: Structure sub-subclass does now initialize correctly
with base class positional arguments.
- Issue #6882: Import uuid creates zombies processes. - Issue #6882: Import uuid creates zombies processes.
- Issue #6635: Fix profiler printing usage message. - Issue #6635: Fix profiler printing usage message.

View File

@ -3936,82 +3936,97 @@ IBUG(char *msg)
return -1; return -1;
} }
/*
This function is called to initialize a Structure or Union with positional
arguments. It calls itself recursively for all Structure or Union base
classes, then retrieves the _fields_ member to associate the argument
position with the correct field name.
Returns -1 on error, or the index of next argument on success.
*/
static int
_init_pos_args(PyObject *self, PyTypeObject *type,
PyObject *args, PyObject *kwds,
int index)
{
StgDictObject *dict;
PyObject *fields;
int i;
if (PyType_stgdict((PyObject *)type->tp_base)) {
index = _init_pos_args(self, type->tp_base,
args, kwds,
index);
if (index == -1)
return -1;
}
dict = PyType_stgdict((PyObject *)type);
fields = PyDict_GetItemString((PyObject *)dict, "_fields_");
if (fields == NULL)
return index;
for (i = 0;
i < dict->length && (i+index) < PyTuple_GET_SIZE(args);
++i) {
PyObject *pair = PySequence_GetItem(fields, i);
PyObject *name, *val;
int res;
if (!pair)
return -1;
name = PySequence_GetItem(pair, 0);
if (!name) {
Py_DECREF(pair);
return -1;
}
val = PyTuple_GET_ITEM(args, i + index);
if (kwds && PyDict_GetItem(kwds, name)) {
char *field = PyBytes_AsString(name);
if (field == NULL) {
PyErr_Clear();
field = "???";
}
PyErr_Format(PyExc_TypeError,
"duplicate values for field '%s'",
field);
Py_DECREF(pair);
Py_DECREF(name);
return -1;
}
res = PyObject_SetAttr(self, name, val);
Py_DECREF(pair);
Py_DECREF(name);
if (res == -1)
return -1;
}
return index + dict->length;
}
static int static int
Struct_init(PyObject *self, PyObject *args, PyObject *kwds) Struct_init(PyObject *self, PyObject *args, PyObject *kwds)
{ {
int i; StgDictObject *stgdict = PyObject_stgdict(self);
PyObject *fields;
/* Optimization possible: Store the attribute names _fields_[x][0] /* Optimization possible: Store the attribute names _fields_[x][0]
* in C accessible fields somewhere ? * in C accessible fields somewhere ?
*/ */
/* Check this code again for correctness! */
if (!PyTuple_Check(args)) { if (!PyTuple_Check(args)) {
PyErr_SetString(PyExc_TypeError, PyErr_SetString(PyExc_TypeError,
"args not a tuple?"); "args not a tuple?");
return -1; return -1;
} }
if (PyTuple_GET_SIZE(args)) { if (PyTuple_GET_SIZE(args)) {
fields = PyObject_GetAttrString(self, "_fields_"); int res = _init_pos_args(self, Py_TYPE(self),
if (!fields) { args, kwds, 0);
PyErr_Clear(); if (res == -1)
fields = PyTuple_New(0); return -1;
if (!fields) if (res < PyTuple_GET_SIZE(args)) {
return -1;
}
if (PyTuple_GET_SIZE(args) > PySequence_Length(fields)) {
Py_DECREF(fields);
PyErr_SetString(PyExc_TypeError, PyErr_SetString(PyExc_TypeError,
"too many initializers"); "too many initializers");
return -1; return -1;
} }
for (i = 0; i < PyTuple_GET_SIZE(args); ++i) {
PyObject *pair = PySequence_GetItem(fields, i);
PyObject *name;
PyObject *val;
if (!pair) {
Py_DECREF(fields);
return IBUG("_fields_[i] failed");
}
name = PySequence_GetItem(pair, 0);
if (!name) {
Py_DECREF(pair);
Py_DECREF(fields);
return IBUG("_fields_[i][0] failed");
}
if (kwds && PyDict_GetItem(kwds, name)) {
char *field = PyBytes_AsString(name);
if (field == NULL) {
PyErr_Clear();
field = "???";
}
PyErr_Format(PyExc_TypeError,
"duplicate values for field %s",
field);
Py_DECREF(pair);
Py_DECREF(name);
Py_DECREF(fields);
return -1;
}
val = PyTuple_GET_ITEM(args, i);
if (-1 == PyObject_SetAttr(self, name, val)) {
Py_DECREF(pair);
Py_DECREF(name);
Py_DECREF(fields);
return -1;
}
Py_DECREF(name);
Py_DECREF(pair);
}
Py_DECREF(fields);
} }
if (kwds) { if (kwds) {