New submission from Jeroen Demeyer <j.deme...@ugent.be>:
NOTE: because of PEP 442, this issue is specific to Python 2. This bug was discovered while adding testcases for bpo-35983 to the Python 2.7 backport. There is a nasty interaction between the trashcan and __del__: if you're very close to the trashcan limit and you're calling __del__, then objects that should have been deallocated in __del__ (in particular, an object involving self) might instead end up in the trashcan. This way, temporary references to self are not cleaned up and self might be resurrected when it shouldn't be. This in turns causes __del__ to be called multiple times. Testcase: class ObjectCounter(object): count = 0 def __init__(self): type(self).count += 1 def __del__(self): L = [self] type(self).count -= 1 L = None for i in range(60): L = [L, ObjectCounter()] del L print(ObjectCounter.count) This is expected to print 0 but in facts it prints -1. There are various ways of fixing this, with varying effectiveness. An obvious solution is bypassing the trashcan completely in __del__. This will deallocate objects correctly but it will cause a stack overflow (on the C level, so crashing Python) if __del__ is called recursively with deep recursion (this is what the trashcan is supposed to prevent). A compromise solution would be lowering the trashcan limit for heap types from 50 to 40: this gives __del__ at least 10 stack frames to work with. Assuming that __del__ code is relatively simple and won't create objects that are too deeply nested, this should work correctly. ---------- components: Interpreter Core messages: 339611 nosy: eric.snow, jdemeyer, matrixise, pitrou, scoder, serhiy.storchaka priority: normal severity: normal status: open title: Trashcan causing duplicated __del__ calls versions: Python 2.7 _______________________________________ Python tracker <rep...@bugs.python.org> <https://bugs.python.org/issue36556> _______________________________________ _______________________________________________ Python-bugs-list mailing list Unsubscribe: https://mail.python.org/mailman/options/python-bugs-list/archive%40mail-archive.com