Thanks to visit codestin.com
Credit goes to github.com

Skip to content

Commit 8fd544f

Browse files
committed
Issue #12791: Break reference cycles early when a generator exits with an exception.
2 parents 18bb330 + a370fcf commit 8fd544f

3 files changed

Lines changed: 76 additions & 0 deletions

File tree

Lib/test/test_exceptions.py

Lines changed: 62 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -608,6 +608,68 @@ def run_gen():
608608
gc_collect()
609609
self.assertEqual(sys.exc_info(), (None, None, None))
610610

611+
def _check_generator_cleanup_exc_state(self, testfunc):
612+
# Issue #12791: exception state is cleaned up as soon as a generator
613+
# is closed (reference cycles are broken).
614+
class MyException(Exception):
615+
def __init__(self, obj):
616+
self.obj = obj
617+
class MyObj:
618+
pass
619+
620+
def raising_gen():
621+
try:
622+
raise MyException(obj)
623+
except MyException:
624+
yield
625+
626+
obj = MyObj()
627+
wr = weakref.ref(obj)
628+
g = raising_gen()
629+
next(g)
630+
testfunc(g)
631+
g = obj = None
632+
obj = wr()
633+
self.assertIs(obj, None)
634+
635+
def test_generator_throw_cleanup_exc_state(self):
636+
def do_throw(g):
637+
try:
638+
g.throw(RuntimeError())
639+
except RuntimeError:
640+
pass
641+
self._check_generator_cleanup_exc_state(do_throw)
642+
643+
def test_generator_close_cleanup_exc_state(self):
644+
def do_close(g):
645+
g.close()
646+
self._check_generator_cleanup_exc_state(do_close)
647+
648+
def test_generator_del_cleanup_exc_state(self):
649+
def do_del(g):
650+
g = None
651+
self._check_generator_cleanup_exc_state(do_del)
652+
653+
def test_generator_next_cleanup_exc_state(self):
654+
def do_next(g):
655+
try:
656+
next(g)
657+
except StopIteration:
658+
pass
659+
else:
660+
self.fail("should have raised StopIteration")
661+
self._check_generator_cleanup_exc_state(do_next)
662+
663+
def test_generator_send_cleanup_exc_state(self):
664+
def do_send(g):
665+
try:
666+
g.send(None)
667+
except StopIteration:
668+
pass
669+
else:
670+
self.fail("should have raised StopIteration")
671+
self._check_generator_cleanup_exc_state(do_send)
672+
611673
def test_3114(self):
612674
# Bug #3114: in its destructor, MyObject retrieves a pointer to
613675
# obsolete and/or deallocated objects.

Misc/NEWS

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,9 @@ What's New in Python 3.3 Alpha 1?
1010
Core and Builtins
1111
-----------------
1212

13+
- Issue #12791: Break reference cycles early when a generator exits with
14+
an exception.
15+
1316
- Issue #12773: Make __doc__ mutable on user-defined classes.
1417

1518
- Issue #12766: Raise an ValueError when creating a class with a class variable

Objects/genobject.c

Lines changed: 11 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -100,6 +100,17 @@ gen_send_ex(PyGenObject *gen, PyObject *arg, int exc)
100100

101101
if (!result || f->f_stacktop == NULL) {
102102
/* generator can't be rerun, so release the frame */
103+
/* first clean reference cycle through stored exception traceback */
104+
PyObject *t, *v, *tb;
105+
t = f->f_exc_type;
106+
v = f->f_exc_value;
107+
tb = f->f_exc_traceback;
108+
f->f_exc_type = NULL;
109+
f->f_exc_value = NULL;
110+
f->f_exc_traceback = NULL;
111+
Py_XDECREF(t);
112+
Py_XDECREF(v);
113+
Py_XDECREF(tb);
103114
Py_DECREF(f);
104115
gen->gi_frame = NULL;
105116
}

0 commit comments

Comments
 (0)