From 6832c81d5d2d730c38bab001e0c1b4b0569dd5ef Mon Sep 17 00:00:00 2001 From: Guido van Rossum Date: Fri, 10 May 2013 08:47:42 -0700 Subject: #17927: Keep frame from referencing cell-ified arguments. --- Lib/test/test_scope.py | 41 ++++++++++++++++++++++++++++++++++++++++- Misc/NEWS | 3 +++ Objects/typeobject.c | 4 ++++ Python/ceval.c | 16 ++++++++++++---- 4 files changed, 59 insertions(+), 5 deletions(-) diff --git a/Lib/test/test_scope.py b/Lib/test/test_scope.py index f4ed244..1c06224 100644 --- a/Lib/test/test_scope.py +++ b/Lib/test/test_scope.py @@ -1,3 +1,4 @@ +import gc import unittest from test.support import check_syntax_error, cpython_only, run_unittest @@ -713,7 +714,6 @@ class ScopeTests(unittest.TestCase): def b(): global a - def testClassNamespaceOverridesClosure(self): # See #17853. x = 42 @@ -727,6 +727,45 @@ class ScopeTests(unittest.TestCase): self.assertFalse(hasattr(X, "x")) self.assertEqual(x, 42) + @cpython_only + def testCellLeak(self): + # Issue 17927. + # + # The issue was that if self was part of a cycle involving the + # frame of a method call, *and* the method contained a nested + # function referencing self, thereby forcing 'self' into a + # cell, setting self to None would not be enough to break the + # frame -- the frame had another reference to the instance, + # which could not be cleared by the code running in the frame + # (though it will be cleared when the frame is collected). + # Without the lambda, setting self to None is enough to break + # the cycle. + logs = [] + class Canary: + def __del__(self): + logs.append('canary') + class Base: + def dig(self): + pass + class Tester(Base): + def __init__(self): + self.canary = Canary() + def dig(self): + if 0: + lambda: self + try: + 1/0 + except Exception as exc: + self.exc = exc + super().dig() + self = None # Break the cycle + tester = Tester() + tester.dig() + del tester + logs.append('collect') + gc.collect() + self.assertEqual(logs, ['canary', 'collect']) + def test_main(): run_unittest(ScopeTests) diff --git a/Misc/NEWS b/Misc/NEWS index d56163d..d807d0b 100644 --- a/Misc/NEWS +++ b/Misc/NEWS @@ -10,6 +10,9 @@ What's New in Python 3.4.0 Alpha 1? Core and Builtins ----------------- +- Issue #17927: Frame objects kept arguments alive if they had been + copied into a cell, even if the cell was cleared. + - Issue #17807: Generators can now be finalized even when they are part of a reference cycle. diff --git a/Objects/typeobject.c b/Objects/typeobject.c index aa67af8..e418a3a 100644 --- a/Objects/typeobject.c +++ b/Objects/typeobject.c @@ -6510,6 +6510,10 @@ 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) { PyErr_SetString(PyExc_RuntimeError, "super(): arg[0] deleted"); diff --git a/Python/ceval.c b/Python/ceval.c index d32b6fb..d6dba56 100644 --- a/Python/ceval.c +++ b/Python/ceval.c @@ -3519,12 +3519,20 @@ PyEval_EvalCodeEx(PyObject *_co, PyObject *globals, PyObject *locals, int arg; /* Possibly account for the cell variable being an argument. */ if (co->co_cell2arg != NULL && - (arg = co->co_cell2arg[i]) != CO_CELL_NOT_AN_ARG) + (arg = co->co_cell2arg[i]) != CO_CELL_NOT_AN_ARG) { c = PyCell_New(GETLOCAL(arg)); - else + if (c == NULL) + goto fail; + /* Reference the cell from the argument slot, for super(). + See typeobject.c. */ + Py_INCREF(c); + SETLOCAL(arg, c); + } + 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) { -- cgit v0.12