2001-01-17 15:11:13 -04:00
|
|
|
from test_support import verify, verbose, TestFailed
|
2001-07-12 10:25:53 -03:00
|
|
|
import sys
|
2000-06-30 02:02:53 -03:00
|
|
|
import gc
|
|
|
|
|
2001-10-02 16:49:47 -03:00
|
|
|
def expect(actual, expected, name):
|
|
|
|
if actual != expected:
|
|
|
|
raise TestFailed, "test_%s: actual %d, expected %d" % (
|
|
|
|
name, actual, expected)
|
|
|
|
|
Add Garbage Collection support to new-style classes (not yet to their
instances).
Also added GC support to various auxiliary types: super, property,
descriptors, wrappers, dictproxy. (Only type objects have a tp_clear
field; the other types are.)
One change was necessary to the GC infrastructure. We have statically
allocated type objects that don't have a GC header (and can't easily
be given one) and heap-allocated type objects that do have a GC
header. Giving these different metatypes would be really ugly: I
tried, and I had to modify pickle.py, cPickle.c, copy.py, add a new
invent a new name for the new metatype and make it a built-in, change
affected tests... In short, a mess. So instead, we add a new type
slot tp_is_gc, which is a simple Boolean function that determines
whether a particular instance has GC headers or not. This slot is
only relevant for types that have the (new) GC flag bit set. If the
tp_is_gc slot is NULL (by far the most common case), all instances of
the type are deemed to have GC headers. This slot is called by the
PyObject_IS_GC() macro (which is only used twice, both times in
gcmodule.c).
I also changed the extern declarations for a bunch of GC-related
functions (_PyObject_GC_Del etc.): these always exist but objimpl.h
only declared them when WITH_CYCLE_GC was defined, but I needed to be
able to reference them without #ifdefs. (When WITH_CYCLE_GC is not
defined, they do the same as their non-GC counterparts anyway.)
2001-10-02 18:24:57 -03:00
|
|
|
def expect_nonzero(actual, name):
|
|
|
|
if actual == 0:
|
|
|
|
raise TestFailed, "test_%s: unexpected zero" % name
|
2001-10-02 16:49:47 -03:00
|
|
|
|
2000-09-22 12:26:20 -03:00
|
|
|
def run_test(name, thunk):
|
|
|
|
if verbose:
|
|
|
|
print "testing %s..." % name,
|
2001-10-02 16:49:47 -03:00
|
|
|
thunk()
|
|
|
|
if verbose:
|
|
|
|
print "ok"
|
2000-09-22 12:26:20 -03:00
|
|
|
|
2000-06-30 02:02:53 -03:00
|
|
|
def test_list():
|
|
|
|
l = []
|
|
|
|
l.append(l)
|
|
|
|
gc.collect()
|
|
|
|
del l
|
2001-10-02 16:49:47 -03:00
|
|
|
expect(gc.collect(), 1, "list")
|
2000-06-30 02:02:53 -03:00
|
|
|
|
|
|
|
def test_dict():
|
|
|
|
d = {}
|
|
|
|
d[1] = d
|
|
|
|
gc.collect()
|
|
|
|
del d
|
2001-10-02 16:49:47 -03:00
|
|
|
expect(gc.collect(), 1, "dict")
|
2000-06-30 02:02:53 -03:00
|
|
|
|
|
|
|
def test_tuple():
|
2000-09-15 19:32:29 -03:00
|
|
|
# since tuples are immutable we close the loop with a list
|
2000-06-30 02:02:53 -03:00
|
|
|
l = []
|
|
|
|
t = (l,)
|
|
|
|
l.append(t)
|
|
|
|
gc.collect()
|
|
|
|
del t
|
|
|
|
del l
|
2001-10-02 16:49:47 -03:00
|
|
|
expect(gc.collect(), 2, "tuple")
|
2000-06-30 02:02:53 -03:00
|
|
|
|
|
|
|
def test_class():
|
|
|
|
class A:
|
|
|
|
pass
|
|
|
|
A.a = A
|
|
|
|
gc.collect()
|
|
|
|
del A
|
Add Garbage Collection support to new-style classes (not yet to their
instances).
Also added GC support to various auxiliary types: super, property,
descriptors, wrappers, dictproxy. (Only type objects have a tp_clear
field; the other types are.)
One change was necessary to the GC infrastructure. We have statically
allocated type objects that don't have a GC header (and can't easily
be given one) and heap-allocated type objects that do have a GC
header. Giving these different metatypes would be really ugly: I
tried, and I had to modify pickle.py, cPickle.c, copy.py, add a new
invent a new name for the new metatype and make it a built-in, change
affected tests... In short, a mess. So instead, we add a new type
slot tp_is_gc, which is a simple Boolean function that determines
whether a particular instance has GC headers or not. This slot is
only relevant for types that have the (new) GC flag bit set. If the
tp_is_gc slot is NULL (by far the most common case), all instances of
the type are deemed to have GC headers. This slot is called by the
PyObject_IS_GC() macro (which is only used twice, both times in
gcmodule.c).
I also changed the extern declarations for a bunch of GC-related
functions (_PyObject_GC_Del etc.): these always exist but objimpl.h
only declared them when WITH_CYCLE_GC was defined, but I needed to be
able to reference them without #ifdefs. (When WITH_CYCLE_GC is not
defined, they do the same as their non-GC counterparts anyway.)
2001-10-02 18:24:57 -03:00
|
|
|
expect_nonzero(gc.collect(), "class")
|
|
|
|
|
2001-10-15 19:49:27 -03:00
|
|
|
def test_newstyleclass():
|
Add Garbage Collection support to new-style classes (not yet to their
instances).
Also added GC support to various auxiliary types: super, property,
descriptors, wrappers, dictproxy. (Only type objects have a tp_clear
field; the other types are.)
One change was necessary to the GC infrastructure. We have statically
allocated type objects that don't have a GC header (and can't easily
be given one) and heap-allocated type objects that do have a GC
header. Giving these different metatypes would be really ugly: I
tried, and I had to modify pickle.py, cPickle.c, copy.py, add a new
invent a new name for the new metatype and make it a built-in, change
affected tests... In short, a mess. So instead, we add a new type
slot tp_is_gc, which is a simple Boolean function that determines
whether a particular instance has GC headers or not. This slot is
only relevant for types that have the (new) GC flag bit set. If the
tp_is_gc slot is NULL (by far the most common case), all instances of
the type are deemed to have GC headers. This slot is called by the
PyObject_IS_GC() macro (which is only used twice, both times in
gcmodule.c).
I also changed the extern declarations for a bunch of GC-related
functions (_PyObject_GC_Del etc.): these always exist but objimpl.h
only declared them when WITH_CYCLE_GC was defined, but I needed to be
able to reference them without #ifdefs. (When WITH_CYCLE_GC is not
defined, they do the same as their non-GC counterparts anyway.)
2001-10-02 18:24:57 -03:00
|
|
|
class A(object):
|
2001-10-15 19:49:27 -03:00
|
|
|
pass
|
Add Garbage Collection support to new-style classes (not yet to their
instances).
Also added GC support to various auxiliary types: super, property,
descriptors, wrappers, dictproxy. (Only type objects have a tp_clear
field; the other types are.)
One change was necessary to the GC infrastructure. We have statically
allocated type objects that don't have a GC header (and can't easily
be given one) and heap-allocated type objects that do have a GC
header. Giving these different metatypes would be really ugly: I
tried, and I had to modify pickle.py, cPickle.c, copy.py, add a new
invent a new name for the new metatype and make it a built-in, change
affected tests... In short, a mess. So instead, we add a new type
slot tp_is_gc, which is a simple Boolean function that determines
whether a particular instance has GC headers or not. This slot is
only relevant for types that have the (new) GC flag bit set. If the
tp_is_gc slot is NULL (by far the most common case), all instances of
the type are deemed to have GC headers. This slot is called by the
PyObject_IS_GC() macro (which is only used twice, both times in
gcmodule.c).
I also changed the extern declarations for a bunch of GC-related
functions (_PyObject_GC_Del etc.): these always exist but objimpl.h
only declared them when WITH_CYCLE_GC was defined, but I needed to be
able to reference them without #ifdefs. (When WITH_CYCLE_GC is not
defined, they do the same as their non-GC counterparts anyway.)
2001-10-02 18:24:57 -03:00
|
|
|
gc.collect()
|
|
|
|
del A
|
|
|
|
expect_nonzero(gc.collect(), "staticclass")
|
|
|
|
|
2000-06-30 02:02:53 -03:00
|
|
|
def test_instance():
|
|
|
|
class A:
|
|
|
|
pass
|
|
|
|
a = A()
|
|
|
|
a.a = a
|
|
|
|
gc.collect()
|
|
|
|
del a
|
Add Garbage Collection support to new-style classes (not yet to their
instances).
Also added GC support to various auxiliary types: super, property,
descriptors, wrappers, dictproxy. (Only type objects have a tp_clear
field; the other types are.)
One change was necessary to the GC infrastructure. We have statically
allocated type objects that don't have a GC header (and can't easily
be given one) and heap-allocated type objects that do have a GC
header. Giving these different metatypes would be really ugly: I
tried, and I had to modify pickle.py, cPickle.c, copy.py, add a new
invent a new name for the new metatype and make it a built-in, change
affected tests... In short, a mess. So instead, we add a new type
slot tp_is_gc, which is a simple Boolean function that determines
whether a particular instance has GC headers or not. This slot is
only relevant for types that have the (new) GC flag bit set. If the
tp_is_gc slot is NULL (by far the most common case), all instances of
the type are deemed to have GC headers. This slot is called by the
PyObject_IS_GC() macro (which is only used twice, both times in
gcmodule.c).
I also changed the extern declarations for a bunch of GC-related
functions (_PyObject_GC_Del etc.): these always exist but objimpl.h
only declared them when WITH_CYCLE_GC was defined, but I needed to be
able to reference them without #ifdefs. (When WITH_CYCLE_GC is not
defined, they do the same as their non-GC counterparts anyway.)
2001-10-02 18:24:57 -03:00
|
|
|
expect_nonzero(gc.collect(), "instance")
|
2000-06-30 02:02:53 -03:00
|
|
|
|
2001-10-05 17:51:39 -03:00
|
|
|
def test_newinstance():
|
|
|
|
class A(object):
|
|
|
|
pass
|
|
|
|
a = A()
|
|
|
|
a.a = a
|
|
|
|
gc.collect()
|
|
|
|
del a
|
|
|
|
expect_nonzero(gc.collect(), "newinstance")
|
|
|
|
class B(list):
|
|
|
|
pass
|
|
|
|
class C(B, A):
|
|
|
|
pass
|
|
|
|
a = C()
|
|
|
|
a.a = a
|
|
|
|
gc.collect()
|
|
|
|
del a
|
|
|
|
expect_nonzero(gc.collect(), "newinstance(2)")
|
|
|
|
|
2000-06-30 02:02:53 -03:00
|
|
|
def test_method():
|
2000-09-15 19:32:29 -03:00
|
|
|
# Tricky: self.__init__ is a bound method, it references the instance.
|
2000-06-30 02:02:53 -03:00
|
|
|
class A:
|
|
|
|
def __init__(self):
|
|
|
|
self.init = self.__init__
|
|
|
|
a = A()
|
|
|
|
gc.collect()
|
|
|
|
del a
|
Add Garbage Collection support to new-style classes (not yet to their
instances).
Also added GC support to various auxiliary types: super, property,
descriptors, wrappers, dictproxy. (Only type objects have a tp_clear
field; the other types are.)
One change was necessary to the GC infrastructure. We have statically
allocated type objects that don't have a GC header (and can't easily
be given one) and heap-allocated type objects that do have a GC
header. Giving these different metatypes would be really ugly: I
tried, and I had to modify pickle.py, cPickle.c, copy.py, add a new
invent a new name for the new metatype and make it a built-in, change
affected tests... In short, a mess. So instead, we add a new type
slot tp_is_gc, which is a simple Boolean function that determines
whether a particular instance has GC headers or not. This slot is
only relevant for types that have the (new) GC flag bit set. If the
tp_is_gc slot is NULL (by far the most common case), all instances of
the type are deemed to have GC headers. This slot is called by the
PyObject_IS_GC() macro (which is only used twice, both times in
gcmodule.c).
I also changed the extern declarations for a bunch of GC-related
functions (_PyObject_GC_Del etc.): these always exist but objimpl.h
only declared them when WITH_CYCLE_GC was defined, but I needed to be
able to reference them without #ifdefs. (When WITH_CYCLE_GC is not
defined, they do the same as their non-GC counterparts anyway.)
2001-10-02 18:24:57 -03:00
|
|
|
expect_nonzero(gc.collect(), "method")
|
2000-06-30 02:02:53 -03:00
|
|
|
|
|
|
|
def test_finalizer():
|
2000-09-15 19:32:29 -03:00
|
|
|
# A() is uncollectable if it is part of a cycle, make sure it shows up
|
|
|
|
# in gc.garbage.
|
2000-06-30 02:02:53 -03:00
|
|
|
class A:
|
|
|
|
def __del__(self): pass
|
|
|
|
class B:
|
|
|
|
pass
|
|
|
|
a = A()
|
|
|
|
a.a = a
|
|
|
|
id_a = id(a)
|
|
|
|
b = B()
|
|
|
|
b.b = b
|
|
|
|
gc.collect()
|
|
|
|
del a
|
|
|
|
del b
|
Add Garbage Collection support to new-style classes (not yet to their
instances).
Also added GC support to various auxiliary types: super, property,
descriptors, wrappers, dictproxy. (Only type objects have a tp_clear
field; the other types are.)
One change was necessary to the GC infrastructure. We have statically
allocated type objects that don't have a GC header (and can't easily
be given one) and heap-allocated type objects that do have a GC
header. Giving these different metatypes would be really ugly: I
tried, and I had to modify pickle.py, cPickle.c, copy.py, add a new
invent a new name for the new metatype and make it a built-in, change
affected tests... In short, a mess. So instead, we add a new type
slot tp_is_gc, which is a simple Boolean function that determines
whether a particular instance has GC headers or not. This slot is
only relevant for types that have the (new) GC flag bit set. If the
tp_is_gc slot is NULL (by far the most common case), all instances of
the type are deemed to have GC headers. This slot is called by the
PyObject_IS_GC() macro (which is only used twice, both times in
gcmodule.c).
I also changed the extern declarations for a bunch of GC-related
functions (_PyObject_GC_Del etc.): these always exist but objimpl.h
only declared them when WITH_CYCLE_GC was defined, but I needed to be
able to reference them without #ifdefs. (When WITH_CYCLE_GC is not
defined, they do the same as their non-GC counterparts anyway.)
2001-10-02 18:24:57 -03:00
|
|
|
expect_nonzero(gc.collect(), "finalizer")
|
2000-09-22 12:26:20 -03:00
|
|
|
for obj in gc.garbage:
|
|
|
|
if id(obj) == id_a:
|
|
|
|
del obj.a
|
|
|
|
break
|
|
|
|
else:
|
2001-10-02 16:49:47 -03:00
|
|
|
raise TestFailed, "didn't find obj in garbage (finalizer)"
|
2000-09-22 12:26:20 -03:00
|
|
|
gc.garbage.remove(obj)
|
2000-06-30 02:02:53 -03:00
|
|
|
|
|
|
|
def test_function():
|
2000-09-15 19:32:29 -03:00
|
|
|
# Tricky: f -> d -> f, code should call d.clear() after the exec to
|
|
|
|
# break the cycle.
|
2000-06-30 02:02:53 -03:00
|
|
|
d = {}
|
|
|
|
exec("def f(): pass\n") in d
|
|
|
|
gc.collect()
|
|
|
|
del d
|
2001-10-02 16:49:47 -03:00
|
|
|
expect(gc.collect(), 2, "function")
|
2000-09-22 12:26:20 -03:00
|
|
|
|
2001-07-12 10:25:53 -03:00
|
|
|
def test_frame():
|
|
|
|
def f():
|
|
|
|
frame = sys._getframe()
|
|
|
|
gc.collect()
|
|
|
|
f()
|
2001-10-02 16:49:47 -03:00
|
|
|
expect(gc.collect(), 1, "frame")
|
2001-07-12 10:25:53 -03:00
|
|
|
|
|
|
|
|
2000-09-22 12:26:20 -03:00
|
|
|
def test_saveall():
|
|
|
|
# Verify that cyclic garbage like lists show up in gc.garbage if the
|
|
|
|
# SAVEALL option is enabled.
|
|
|
|
debug = gc.get_debug()
|
|
|
|
gc.set_debug(debug | gc.DEBUG_SAVEALL)
|
|
|
|
l = []
|
|
|
|
l.append(l)
|
|
|
|
id_l = id(l)
|
|
|
|
del l
|
|
|
|
gc.collect()
|
|
|
|
try:
|
|
|
|
for obj in gc.garbage:
|
|
|
|
if id(obj) == id_l:
|
|
|
|
del obj[:]
|
|
|
|
break
|
|
|
|
else:
|
2001-10-02 16:49:47 -03:00
|
|
|
raise TestFailed, "didn't find obj in garbage (saveall)"
|
2000-09-22 12:26:20 -03:00
|
|
|
gc.garbage.remove(obj)
|
|
|
|
finally:
|
|
|
|
gc.set_debug(debug)
|
2000-06-30 02:02:53 -03:00
|
|
|
|
2000-09-15 19:32:29 -03:00
|
|
|
def test_del():
|
|
|
|
# __del__ methods can trigger collection, make this to happen
|
|
|
|
thresholds = gc.get_threshold()
|
|
|
|
gc.enable()
|
|
|
|
gc.set_threshold(1)
|
|
|
|
|
2000-10-23 14:22:08 -03:00
|
|
|
class A:
|
|
|
|
def __del__(self):
|
|
|
|
dir(self)
|
2000-09-15 19:32:29 -03:00
|
|
|
a = A()
|
|
|
|
del a
|
|
|
|
|
|
|
|
gc.disable()
|
|
|
|
apply(gc.set_threshold, thresholds)
|
2000-10-23 14:22:08 -03:00
|
|
|
|
2002-03-28 17:08:30 -04:00
|
|
|
class Ouch:
|
|
|
|
n = 0
|
|
|
|
def __del__(self):
|
|
|
|
Ouch.n = Ouch.n + 1
|
|
|
|
if Ouch.n % 7 == 0:
|
|
|
|
gc.collect()
|
|
|
|
|
|
|
|
def test_trashcan():
|
|
|
|
# "trashcan" is a hack to prevent stack overflow when deallocating
|
|
|
|
# very deeply nested tuples etc. It works in part by abusing the
|
|
|
|
# type pointer and refcount fields, and that can yield horrible
|
|
|
|
# problems when gc tries to traverse the structures.
|
|
|
|
# If this test fails (as it does in 2.0, 2.1 and 2.2), it will
|
|
|
|
# most likely die via segfault.
|
|
|
|
|
|
|
|
gc.enable()
|
|
|
|
N = 200
|
2002-03-28 17:22:25 -04:00
|
|
|
for count in range(3):
|
2002-03-28 17:08:30 -04:00
|
|
|
t = []
|
|
|
|
for i in range(N):
|
|
|
|
t = [t, Ouch()]
|
|
|
|
u = []
|
|
|
|
for i in range(N):
|
|
|
|
u = [u, Ouch()]
|
|
|
|
v = {}
|
|
|
|
for i in range(N):
|
|
|
|
v = {1: v, 2: Ouch()}
|
|
|
|
gc.disable()
|
2000-06-30 02:02:53 -03:00
|
|
|
|
|
|
|
def test_all():
|
2001-10-02 16:49:47 -03:00
|
|
|
gc.collect() # Delete 2nd generation garbage
|
2000-09-22 12:26:20 -03:00
|
|
|
run_test("lists", test_list)
|
|
|
|
run_test("dicts", test_dict)
|
|
|
|
run_test("tuples", test_tuple)
|
|
|
|
run_test("classes", test_class)
|
2001-10-15 19:49:27 -03:00
|
|
|
run_test("new style classes", test_newstyleclass)
|
2000-09-22 12:26:20 -03:00
|
|
|
run_test("instances", test_instance)
|
2001-10-05 17:51:39 -03:00
|
|
|
run_test("new instances", test_newinstance)
|
2000-09-22 12:26:20 -03:00
|
|
|
run_test("methods", test_method)
|
|
|
|
run_test("functions", test_function)
|
2001-07-12 10:25:53 -03:00
|
|
|
run_test("frames", test_frame)
|
2000-09-22 12:26:20 -03:00
|
|
|
run_test("finalizers", test_finalizer)
|
|
|
|
run_test("__del__", test_del)
|
|
|
|
run_test("saveall", test_saveall)
|
2002-03-28 17:08:30 -04:00
|
|
|
run_test("trashcan", test_trashcan)
|
2000-09-22 12:26:20 -03:00
|
|
|
|
|
|
|
def test():
|
|
|
|
if verbose:
|
|
|
|
print "disabling automatic collection"
|
2000-08-06 19:45:31 -03:00
|
|
|
enabled = gc.isenabled()
|
|
|
|
gc.disable()
|
2001-01-17 15:11:13 -04:00
|
|
|
verify(not gc.isenabled() )
|
2000-09-22 12:26:20 -03:00
|
|
|
debug = gc.get_debug()
|
|
|
|
gc.set_debug(debug & ~gc.DEBUG_LEAK) # this test is supposed to leak
|
|
|
|
|
|
|
|
try:
|
|
|
|
test_all()
|
|
|
|
finally:
|
|
|
|
gc.set_debug(debug)
|
|
|
|
# test gc.enable() even if GC is disabled by default
|
|
|
|
if verbose:
|
|
|
|
print "restoring automatic collection"
|
|
|
|
# make sure to always test gc.enable()
|
|
|
|
gc.enable()
|
2001-01-17 15:11:13 -04:00
|
|
|
verify(gc.isenabled())
|
2000-09-22 12:26:20 -03:00
|
|
|
if not enabled:
|
|
|
|
gc.disable()
|
|
|
|
|
|
|
|
|
|
|
|
test()
|