summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorBenjamin Peterson <benjamin@python.org>2013-05-12 23:16:06 (GMT)
committerBenjamin Peterson <benjamin@python.org>2013-05-12 23:16:06 (GMT)
commit159ae41da68ff1f1d85ed66fa6410bd9ba62ed5b (patch)
tree379d3664706b0a0dcc7d32f762d88595ae8e84bf
parent3bfc5f5d833f081089e181cadf52d4ec50e62d13 (diff)
downloadcpython-159ae41da68ff1f1d85ed66fa6410bd9ba62ed5b.zip
cpython-159ae41da68ff1f1d85ed66fa6410bd9ba62ed5b.tar.gz
cpython-159ae41da68ff1f1d85ed66fa6410bd9ba62ed5b.tar.bz2
when an argument is a cell, set the local copy to NULL (see #17927)
-rw-r--r--Lib/test/test_super.py13
-rw-r--r--Objects/typeobject.c14
-rw-r--r--Python/ceval.c12
3 files changed, 28 insertions, 11 deletions
diff --git a/Lib/test/test_super.py b/Lib/test/test_super.py
index f6469cf..1e272ee 100644
--- a/Lib/test/test_super.py
+++ b/Lib/test/test_super.py
@@ -130,6 +130,19 @@ class TestSuper(unittest.TestCase):
super()
self.assertRaises(RuntimeError, X().f)
+ def test_cell_as_self(self):
+ class X:
+ def meth(self):
+ super()
+
+ def f():
+ k = X()
+ def g():
+ return k
+ return g
+ c = f().__closure__[0]
+ self.assertRaises(TypeError, X.meth, c)
+
def test_main():
support.run_unittest(TestSuper)
diff --git a/Objects/typeobject.c b/Objects/typeobject.c
index e418a3a..a351667 100644
--- a/Objects/typeobject.c
+++ b/Objects/typeobject.c
@@ -6510,9 +6510,17 @@ super_init(PyObject *self, PyObject *args, PyObject *kwds)
return -1;
}
obj = f->f_localsplus[0];
- if (obj != NULL && PyCell_Check(obj)) {
- /* It might be a cell. See cell var initialization in ceval.c. */
- obj = PyCell_GET(obj);
+ if (obj == NULL && co->co_cell2arg) {
+ /* The first argument might be a cell. */
+ n = PyTuple_GET_SIZE(co->co_cellvars);
+ for (i = 0; i < n; i++) {
+ if (co->co_cell2arg[i] == 0) {
+ PyObject *cell = f->f_localsplus[co->co_nlocals + i];
+ assert(PyCell_Check(cell));
+ obj = PyCell_GET(cell);
+ break;
+ }
+ }
}
if (obj == NULL) {
PyErr_SetString(PyExc_RuntimeError,
diff --git a/Python/ceval.c b/Python/ceval.c
index d6dba56..e211e4f 100644
--- a/Python/ceval.c
+++ b/Python/ceval.c
@@ -3521,18 +3521,14 @@ PyEval_EvalCodeEx(PyObject *_co, PyObject *globals, PyObject *locals,
if (co->co_cell2arg != NULL &&
(arg = co->co_cell2arg[i]) != CO_CELL_NOT_AN_ARG) {
c = PyCell_New(GETLOCAL(arg));
- if (c == NULL)
- goto fail;
- /* Reference the cell from the argument slot, for super().
- See typeobject.c. */
- Py_INCREF(c);
- SETLOCAL(arg, c);
+ /* Clear the local copy. */
+ SETLOCAL(arg, NULL);
}
else {
c = PyCell_New(NULL);
- if (c == NULL)
- goto fail;
}
+ if (c == NULL)
+ goto fail;
SETLOCAL(co->co_nlocals + i, c);
}
for (i = 0; i < PyTuple_GET_SIZE(co->co_freevars); ++i) {